| Guido van Rossum | e7b146f | 2000-02-04 15:28:42 +0000 | [diff] [blame] | 1 | """Random variable generators. | 
| Guido van Rossum | ff03b1a | 1994-03-09 12:55:02 +0000 | [diff] [blame] | 2 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 3 |     integers | 
 | 4 |     -------- | 
 | 5 |            uniform within range | 
 | 6 |  | 
 | 7 |     sequences | 
 | 8 |     --------- | 
 | 9 |            pick random element | 
| Raymond Hettinger | f24eb35 | 2002-11-12 17:41:57 +0000 | [diff] [blame] | 10 |            pick random sample | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 11 |            generate random permutation | 
 | 12 |  | 
| Guido van Rossum | e7b146f | 2000-02-04 15:28:42 +0000 | [diff] [blame] | 13 |     distributions on the real line: | 
 | 14 |     ------------------------------ | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 15 |            uniform | 
| Raymond Hettinger | bbc50ea | 2008-03-23 13:32:32 +0000 | [diff] [blame] | 16 |            triangular | 
| Guido van Rossum | e7b146f | 2000-02-04 15:28:42 +0000 | [diff] [blame] | 17 |            normal (Gaussian) | 
 | 18 |            lognormal | 
 | 19 |            negative exponential | 
 | 20 |            gamma | 
 | 21 |            beta | 
| Raymond Hettinger | 40f6217 | 2002-12-29 23:03:38 +0000 | [diff] [blame] | 22 |            pareto | 
 | 23 |            Weibull | 
| Guido van Rossum | ff03b1a | 1994-03-09 12:55:02 +0000 | [diff] [blame] | 24 |  | 
| Guido van Rossum | e7b146f | 2000-02-04 15:28:42 +0000 | [diff] [blame] | 25 |     distributions on the circle (angles 0 to 2pi) | 
 | 26 |     --------------------------------------------- | 
 | 27 |            circular uniform | 
 | 28 |            von Mises | 
 | 29 |  | 
| Raymond Hettinger | 40f6217 | 2002-12-29 23:03:38 +0000 | [diff] [blame] | 30 | General notes on the underlying Mersenne Twister core generator: | 
| Guido van Rossum | e7b146f | 2000-02-04 15:28:42 +0000 | [diff] [blame] | 31 |  | 
| Raymond Hettinger | 40f6217 | 2002-12-29 23:03:38 +0000 | [diff] [blame] | 32 | * The period is 2**19937-1. | 
| Tim Peters | 0e11595 | 2006-06-10 22:51:45 +0000 | [diff] [blame] | 33 | * It is one of the most extensively tested generators in existence. | 
 | 34 | * Without a direct way to compute N steps forward, the semantics of | 
 | 35 |   jumpahead(n) are weakened to simply jump to another distant state and rely | 
 | 36 |   on the large period to avoid overlapping sequences. | 
 | 37 | * The random() method is implemented in C, executes in a single Python step, | 
 | 38 |   and is, therefore, threadsafe. | 
| Tim Peters | e360d95 | 2001-01-26 10:00:39 +0000 | [diff] [blame] | 39 |  | 
| Guido van Rossum | e7b146f | 2000-02-04 15:28:42 +0000 | [diff] [blame] | 40 | """ | 
| Guido van Rossum | d03e119 | 1998-05-29 17:51:31 +0000 | [diff] [blame] | 41 |  | 
| Raymond Hettinger | c4f7bab | 2008-03-23 19:37:53 +0000 | [diff] [blame] | 42 | from __future__ import division | 
| Raymond Hettinger | 2f726e9 | 2003-10-05 09:09:15 +0000 | [diff] [blame] | 43 | from warnings import warn as _warn | 
 | 44 | from types import MethodType as _MethodType, BuiltinMethodType as _BuiltinMethodType | 
| Raymond Hettinger | 91e27c2 | 2005-08-19 01:36:35 +0000 | [diff] [blame] | 45 | from math import log as _log, exp as _exp, pi as _pi, e as _e, ceil as _ceil | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 46 | from math import sqrt as _sqrt, acos as _acos, cos as _cos, sin as _sin | 
| Raymond Hettinger | c1c43ca | 2004-09-05 00:00:42 +0000 | [diff] [blame] | 47 | from os import urandom as _urandom | 
 | 48 | from binascii import hexlify as _hexlify | 
| Guido van Rossum | ff03b1a | 1994-03-09 12:55:02 +0000 | [diff] [blame] | 49 |  | 
| Raymond Hettinger | f24eb35 | 2002-11-12 17:41:57 +0000 | [diff] [blame] | 50 | __all__ = ["Random","seed","random","uniform","randint","choice","sample", | 
| Skip Montanaro | 0de6580 | 2001-02-15 22:15:14 +0000 | [diff] [blame] | 51 |            "randrange","shuffle","normalvariate","lognormvariate", | 
| Raymond Hettinger | bbc50ea | 2008-03-23 13:32:32 +0000 | [diff] [blame] | 52 |            "expovariate","vonmisesvariate","gammavariate","triangular", | 
| Raymond Hettinger | f8a52d3 | 2003-08-05 12:23:19 +0000 | [diff] [blame] | 53 |            "gauss","betavariate","paretovariate","weibullvariate", | 
| Raymond Hettinger | 356a459 | 2004-08-30 06:14:31 +0000 | [diff] [blame] | 54 |            "getstate","setstate","jumpahead", "WichmannHill", "getrandbits", | 
| Raymond Hettinger | 23f1241 | 2004-09-13 22:23:21 +0000 | [diff] [blame] | 55 |            "SystemRandom"] | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 56 |  | 
 | 57 | NV_MAGICCONST = 4 * _exp(-0.5)/_sqrt(2.0) | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 58 | TWOPI = 2.0*_pi | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 59 | LOG4 = _log(4.0) | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 60 | SG_MAGICCONST = 1.0 + _log(4.5) | 
| Raymond Hettinger | 2f726e9 | 2003-10-05 09:09:15 +0000 | [diff] [blame] | 61 | BPF = 53        # Number of bits in a float | 
| Tim Peters | 7c2a85b | 2004-08-31 02:19:55 +0000 | [diff] [blame] | 62 | RECIP_BPF = 2**-BPF | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 63 |  | 
| Raymond Hettinger | 356a459 | 2004-08-30 06:14:31 +0000 | [diff] [blame] | 64 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 65 | # Translated by Guido van Rossum from C source provided by | 
| Raymond Hettinger | 40f6217 | 2002-12-29 23:03:38 +0000 | [diff] [blame] | 66 | # Adrian Baddeley.  Adapted by Raymond Hettinger for use with | 
| Raymond Hettinger | 3fa19d7 | 2004-08-31 01:05:15 +0000 | [diff] [blame] | 67 | # the Mersenne Twister  and os.urandom() core generators. | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 68 |  | 
| Raymond Hettinger | 145a4a0 | 2003-01-07 10:25:55 +0000 | [diff] [blame] | 69 | import _random | 
| Raymond Hettinger | 40f6217 | 2002-12-29 23:03:38 +0000 | [diff] [blame] | 70 |  | 
| Raymond Hettinger | 145a4a0 | 2003-01-07 10:25:55 +0000 | [diff] [blame] | 71 | class Random(_random.Random): | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 72 |     """Random number generator base class used by bound module functions. | 
 | 73 |  | 
 | 74 |     Used to instantiate instances of Random to get generators that don't | 
 | 75 |     share state.  Especially useful for multi-threaded programs, creating | 
 | 76 |     a different instance of Random for each thread, and using the jumpahead() | 
 | 77 |     method to ensure that the generated sequences seen by each thread don't | 
 | 78 |     overlap. | 
 | 79 |  | 
 | 80 |     Class Random can also be subclassed if you want to use a different basic | 
 | 81 |     generator of your own devising: in that case, override the following | 
| Benjamin Peterson | f2eb2b4 | 2008-07-30 13:46:53 +0000 | [diff] [blame] | 82 |     methods: random(), seed(), getstate(), setstate() and jumpahead(). | 
 | 83 |     Optionally, implement a getrandbits() method so that randrange() can cover | 
 | 84 |     arbitrarily large ranges. | 
| Raymond Hettinger | ef4d4bd | 2002-05-23 23:58:17 +0000 | [diff] [blame] | 85 |  | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 86 |     """ | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 87 |  | 
| Martin v. Löwis | 6b449f4 | 2007-12-03 19:20:02 +0000 | [diff] [blame] | 88 |     VERSION = 3     # used by getstate/setstate | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 89 |  | 
 | 90 |     def __init__(self, x=None): | 
 | 91 |         """Initialize an instance. | 
 | 92 |  | 
 | 93 |         Optional argument x controls seeding, as for Random.seed(). | 
 | 94 |         """ | 
 | 95 |  | 
 | 96 |         self.seed(x) | 
| Raymond Hettinger | 40f6217 | 2002-12-29 23:03:38 +0000 | [diff] [blame] | 97 |         self.gauss_next = None | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 98 |  | 
| Tim Peters | 0de88fc | 2001-02-01 04:59:18 +0000 | [diff] [blame] | 99 |     def seed(self, a=None): | 
 | 100 |         """Initialize internal state from hashable object. | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 101 |  | 
| Raymond Hettinger | 23f1241 | 2004-09-13 22:23:21 +0000 | [diff] [blame] | 102 |         None or no argument seeds from current time or from an operating | 
 | 103 |         system specific randomness source if available. | 
| Tim Peters | 0de88fc | 2001-02-01 04:59:18 +0000 | [diff] [blame] | 104 |  | 
| Tim Peters | bcd725f | 2001-02-01 10:06:53 +0000 | [diff] [blame] | 105 |         If a is not None or an int or long, hash(a) is used instead. | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 106 |         """ | 
 | 107 |  | 
| Raymond Hettinger | 3081d59 | 2003-08-09 18:30:57 +0000 | [diff] [blame] | 108 |         if a is None: | 
| Raymond Hettinger | c1c43ca | 2004-09-05 00:00:42 +0000 | [diff] [blame] | 109 |             try: | 
 | 110 |                 a = long(_hexlify(_urandom(16)), 16) | 
 | 111 |             except NotImplementedError: | 
| Raymond Hettinger | 356a459 | 2004-08-30 06:14:31 +0000 | [diff] [blame] | 112 |                 import time | 
 | 113 |                 a = long(time.time() * 256) # use fractional seconds | 
| Raymond Hettinger | 356a459 | 2004-08-30 06:14:31 +0000 | [diff] [blame] | 114 |  | 
| Raymond Hettinger | 145a4a0 | 2003-01-07 10:25:55 +0000 | [diff] [blame] | 115 |         super(Random, self).seed(a) | 
| Tim Peters | 46c04e1 | 2002-05-05 20:40:00 +0000 | [diff] [blame] | 116 |         self.gauss_next = None | 
 | 117 |  | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 118 |     def getstate(self): | 
 | 119 |         """Return internal state; can be passed to setstate() later.""" | 
| Raymond Hettinger | 145a4a0 | 2003-01-07 10:25:55 +0000 | [diff] [blame] | 120 |         return self.VERSION, super(Random, self).getstate(), self.gauss_next | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 121 |  | 
 | 122 |     def setstate(self, state): | 
 | 123 |         """Restore internal state from object returned by getstate().""" | 
 | 124 |         version = state[0] | 
| Martin v. Löwis | 6b449f4 | 2007-12-03 19:20:02 +0000 | [diff] [blame] | 125 |         if version == 3: | 
| Raymond Hettinger | 40f6217 | 2002-12-29 23:03:38 +0000 | [diff] [blame] | 126 |             version, internalstate, self.gauss_next = state | 
| Raymond Hettinger | 145a4a0 | 2003-01-07 10:25:55 +0000 | [diff] [blame] | 127 |             super(Random, self).setstate(internalstate) | 
| Martin v. Löwis | 6b449f4 | 2007-12-03 19:20:02 +0000 | [diff] [blame] | 128 |         elif version == 2: | 
 | 129 |             version, internalstate, self.gauss_next = state | 
 | 130 |             # In version 2, the state was saved as signed ints, which causes | 
 | 131 |             #   inconsistencies between 32/64-bit systems. The state is | 
 | 132 |             #   really unsigned 32-bit ints, so we convert negative ints from | 
 | 133 |             #   version 2 to positive longs for version 3. | 
 | 134 |             try: | 
 | 135 |                 internalstate = tuple( long(x) % (2**32) for x in internalstate ) | 
 | 136 |             except ValueError, e: | 
 | 137 |                 raise TypeError, e | 
 | 138 |             super(Random, self).setstate(internalstate) | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 139 |         else: | 
 | 140 |             raise ValueError("state with version %s passed to " | 
 | 141 |                              "Random.setstate() of version %s" % | 
 | 142 |                              (version, self.VERSION)) | 
 | 143 |  | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 144 | ## ---- Methods below this point do not need to be overridden when | 
 | 145 | ## ---- subclassing for the purpose of using a different core generator. | 
 | 146 |  | 
 | 147 | ## -------------------- pickle support  ------------------- | 
 | 148 |  | 
 | 149 |     def __getstate__(self): # for pickle | 
 | 150 |         return self.getstate() | 
 | 151 |  | 
 | 152 |     def __setstate__(self, state):  # for pickle | 
 | 153 |         self.setstate(state) | 
 | 154 |  | 
| Raymond Hettinger | 5f078ff | 2003-06-24 20:29:04 +0000 | [diff] [blame] | 155 |     def __reduce__(self): | 
 | 156 |         return self.__class__, (), self.getstate() | 
 | 157 |  | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 158 | ## -------------------- integer methods  ------------------- | 
 | 159 |  | 
| Raymond Hettinger | 2f726e9 | 2003-10-05 09:09:15 +0000 | [diff] [blame] | 160 |     def randrange(self, start, stop=None, step=1, int=int, default=None, | 
 | 161 |                   maxwidth=1L<<BPF): | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 162 |         """Choose a random item from range(start, stop[, step]). | 
 | 163 |  | 
 | 164 |         This fixes the problem with randint() which includes the | 
 | 165 |         endpoint; in Python this is usually not what you want. | 
| Raymond Hettinger | 2f726e9 | 2003-10-05 09:09:15 +0000 | [diff] [blame] | 166 |         Do not supply the 'int', 'default', and 'maxwidth' arguments. | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 167 |         """ | 
 | 168 |  | 
 | 169 |         # This code is a bit messy to make it fast for the | 
| Tim Peters | 9146f27 | 2002-08-16 03:41:39 +0000 | [diff] [blame] | 170 |         # common case while still doing adequate error checking. | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 171 |         istart = int(start) | 
 | 172 |         if istart != start: | 
 | 173 |             raise ValueError, "non-integer arg 1 for randrange()" | 
 | 174 |         if stop is default: | 
 | 175 |             if istart > 0: | 
| Raymond Hettinger | 2f726e9 | 2003-10-05 09:09:15 +0000 | [diff] [blame] | 176 |                 if istart >= maxwidth: | 
 | 177 |                     return self._randbelow(istart) | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 178 |                 return int(self.random() * istart) | 
 | 179 |             raise ValueError, "empty range for randrange()" | 
| Tim Peters | 9146f27 | 2002-08-16 03:41:39 +0000 | [diff] [blame] | 180 |  | 
 | 181 |         # stop argument supplied. | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 182 |         istop = int(stop) | 
 | 183 |         if istop != stop: | 
 | 184 |             raise ValueError, "non-integer stop for randrange()" | 
| Raymond Hettinger | 2f726e9 | 2003-10-05 09:09:15 +0000 | [diff] [blame] | 185 |         width = istop - istart | 
 | 186 |         if step == 1 and width > 0: | 
| Tim Peters | 76ca1d4 | 2003-06-19 03:46:46 +0000 | [diff] [blame] | 187 |             # Note that | 
| Raymond Hettinger | 2f726e9 | 2003-10-05 09:09:15 +0000 | [diff] [blame] | 188 |             #     int(istart + self.random()*width) | 
| Tim Peters | 76ca1d4 | 2003-06-19 03:46:46 +0000 | [diff] [blame] | 189 |             # instead would be incorrect.  For example, consider istart | 
 | 190 |             # = -2 and istop = 0.  Then the guts would be in | 
 | 191 |             # -2.0 to 0.0 exclusive on both ends (ignoring that random() | 
 | 192 |             # might return 0.0), and because int() truncates toward 0, the | 
 | 193 |             # final result would be -1 or 0 (instead of -2 or -1). | 
| Raymond Hettinger | 2f726e9 | 2003-10-05 09:09:15 +0000 | [diff] [blame] | 194 |             #     istart + int(self.random()*width) | 
| Tim Peters | 76ca1d4 | 2003-06-19 03:46:46 +0000 | [diff] [blame] | 195 |             # would also be incorrect, for a subtler reason:  the RHS | 
 | 196 |             # can return a long, and then randrange() would also return | 
 | 197 |             # a long, but we're supposed to return an int (for backward | 
 | 198 |             # compatibility). | 
| Raymond Hettinger | 2f726e9 | 2003-10-05 09:09:15 +0000 | [diff] [blame] | 199 |  | 
 | 200 |             if width >= maxwidth: | 
| Tim Peters | 58eb11c | 2004-01-18 20:29:55 +0000 | [diff] [blame] | 201 |                 return int(istart + self._randbelow(width)) | 
| Raymond Hettinger | 2f726e9 | 2003-10-05 09:09:15 +0000 | [diff] [blame] | 202 |             return int(istart + int(self.random()*width)) | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 203 |         if step == 1: | 
| Raymond Hettinger | 2f726e9 | 2003-10-05 09:09:15 +0000 | [diff] [blame] | 204 |             raise ValueError, "empty range for randrange() (%d,%d, %d)" % (istart, istop, width) | 
| Tim Peters | 9146f27 | 2002-08-16 03:41:39 +0000 | [diff] [blame] | 205 |  | 
 | 206 |         # Non-unit step argument supplied. | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 207 |         istep = int(step) | 
 | 208 |         if istep != step: | 
 | 209 |             raise ValueError, "non-integer step for randrange()" | 
 | 210 |         if istep > 0: | 
| Raymond Hettinger | ffdb8bb | 2004-09-27 15:29:05 +0000 | [diff] [blame] | 211 |             n = (width + istep - 1) // istep | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 212 |         elif istep < 0: | 
| Raymond Hettinger | ffdb8bb | 2004-09-27 15:29:05 +0000 | [diff] [blame] | 213 |             n = (width + istep + 1) // istep | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 214 |         else: | 
 | 215 |             raise ValueError, "zero step for randrange()" | 
 | 216 |  | 
 | 217 |         if n <= 0: | 
 | 218 |             raise ValueError, "empty range for randrange()" | 
| Raymond Hettinger | 2f726e9 | 2003-10-05 09:09:15 +0000 | [diff] [blame] | 219 |  | 
 | 220 |         if n >= maxwidth: | 
| Raymond Hettinger | 94547f7 | 2006-12-20 06:42:06 +0000 | [diff] [blame] | 221 |             return istart + istep*self._randbelow(n) | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 222 |         return istart + istep*int(self.random() * n) | 
 | 223 |  | 
 | 224 |     def randint(self, a, b): | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 225 |         """Return random integer in range [a, b], including both end points. | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 226 |         """ | 
 | 227 |  | 
 | 228 |         return self.randrange(a, b+1) | 
 | 229 |  | 
| Raymond Hettinger | 2f726e9 | 2003-10-05 09:09:15 +0000 | [diff] [blame] | 230 |     def _randbelow(self, n, _log=_log, int=int, _maxwidth=1L<<BPF, | 
 | 231 |                    _Method=_MethodType, _BuiltinMethod=_BuiltinMethodType): | 
 | 232 |         """Return a random int in the range [0,n) | 
 | 233 |  | 
 | 234 |         Handles the case where n has more bits than returned | 
 | 235 |         by a single call to the underlying generator. | 
 | 236 |         """ | 
 | 237 |  | 
 | 238 |         try: | 
 | 239 |             getrandbits = self.getrandbits | 
 | 240 |         except AttributeError: | 
 | 241 |             pass | 
 | 242 |         else: | 
 | 243 |             # Only call self.getrandbits if the original random() builtin method | 
 | 244 |             # has not been overridden or if a new getrandbits() was supplied. | 
 | 245 |             # This assures that the two methods correspond. | 
 | 246 |             if type(self.random) is _BuiltinMethod or type(getrandbits) is _Method: | 
 | 247 |                 k = int(1.00001 + _log(n-1, 2.0))   # 2**k > n-1 > 2**(k-2) | 
 | 248 |                 r = getrandbits(k) | 
 | 249 |                 while r >= n: | 
 | 250 |                     r = getrandbits(k) | 
 | 251 |                 return r | 
 | 252 |         if n >= _maxwidth: | 
 | 253 |             _warn("Underlying random() generator does not supply \n" | 
 | 254 |                 "enough bits to choose from a population range this large") | 
 | 255 |         return int(self.random() * n) | 
 | 256 |  | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 257 | ## -------------------- sequence methods  ------------------- | 
 | 258 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 259 |     def choice(self, seq): | 
 | 260 |         """Choose a random element from a non-empty sequence.""" | 
| Raymond Hettinger | 5dae505 | 2004-06-07 02:07:15 +0000 | [diff] [blame] | 261 |         return seq[int(self.random() * len(seq))]  # raises IndexError if seq is empty | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 262 |  | 
 | 263 |     def shuffle(self, x, random=None, int=int): | 
 | 264 |         """x, random=random.random -> shuffle list x in place; return None. | 
 | 265 |  | 
 | 266 |         Optional arg random is a 0-argument function returning a random | 
 | 267 |         float in [0.0, 1.0); by default, the standard random.random. | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 268 |         """ | 
 | 269 |  | 
 | 270 |         if random is None: | 
 | 271 |             random = self.random | 
| Raymond Hettinger | 85c20a4 | 2003-11-06 14:06:48 +0000 | [diff] [blame] | 272 |         for i in reversed(xrange(1, len(x))): | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 273 |             # pick an element in x[:i+1] with which to exchange x[i] | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 274 |             j = int(random() * (i+1)) | 
 | 275 |             x[i], x[j] = x[j], x[i] | 
 | 276 |  | 
| Raymond Hettinger | fdbe522 | 2003-06-13 07:01:51 +0000 | [diff] [blame] | 277 |     def sample(self, population, k): | 
| Raymond Hettinger | f24eb35 | 2002-11-12 17:41:57 +0000 | [diff] [blame] | 278 |         """Chooses k unique random elements from a population sequence. | 
 | 279 |  | 
| Raymond Hettinger | c0b4034 | 2002-11-13 15:26:37 +0000 | [diff] [blame] | 280 |         Returns a new list containing elements from the population while | 
 | 281 |         leaving the original population unchanged.  The resulting list is | 
 | 282 |         in selection order so that all sub-slices will also be valid random | 
 | 283 |         samples.  This allows raffle winners (the sample) to be partitioned | 
 | 284 |         into grand prize and second place winners (the subslices). | 
| Raymond Hettinger | f24eb35 | 2002-11-12 17:41:57 +0000 | [diff] [blame] | 285 |  | 
| Raymond Hettinger | c0b4034 | 2002-11-13 15:26:37 +0000 | [diff] [blame] | 286 |         Members of the population need not be hashable or unique.  If the | 
 | 287 |         population contains repeats, then each occurrence is a possible | 
 | 288 |         selection in the sample. | 
| Raymond Hettinger | f24eb35 | 2002-11-12 17:41:57 +0000 | [diff] [blame] | 289 |  | 
| Raymond Hettinger | c0b4034 | 2002-11-13 15:26:37 +0000 | [diff] [blame] | 290 |         To choose a sample in a range of integers, use xrange as an argument. | 
 | 291 |         This is especially fast and space efficient for sampling from a | 
 | 292 |         large population:   sample(xrange(10000000), 60) | 
| Raymond Hettinger | f24eb35 | 2002-11-12 17:41:57 +0000 | [diff] [blame] | 293 |         """ | 
 | 294 |  | 
| Raymond Hettinger | c0b4034 | 2002-11-13 15:26:37 +0000 | [diff] [blame] | 295 |         # Sampling without replacement entails tracking either potential | 
| Raymond Hettinger | 91e27c2 | 2005-08-19 01:36:35 +0000 | [diff] [blame] | 296 |         # selections (the pool) in a list or previous selections in a set. | 
| Raymond Hettinger | c0b4034 | 2002-11-13 15:26:37 +0000 | [diff] [blame] | 297 |  | 
| Jeremy Hylton | 2b55d35 | 2004-02-23 17:27:57 +0000 | [diff] [blame] | 298 |         # When the number of selections is small compared to the | 
 | 299 |         # population, then tracking selections is efficient, requiring | 
| Raymond Hettinger | 91e27c2 | 2005-08-19 01:36:35 +0000 | [diff] [blame] | 300 |         # only a small set and an occasional reselection.  For | 
| Jeremy Hylton | 2b55d35 | 2004-02-23 17:27:57 +0000 | [diff] [blame] | 301 |         # a larger number of selections, the pool tracking method is | 
 | 302 |         # preferred since the list takes less space than the | 
| Raymond Hettinger | 91e27c2 | 2005-08-19 01:36:35 +0000 | [diff] [blame] | 303 |         # set and it doesn't suffer from frequent reselections. | 
| Raymond Hettinger | c0b4034 | 2002-11-13 15:26:37 +0000 | [diff] [blame] | 304 |  | 
| Raymond Hettinger | f24eb35 | 2002-11-12 17:41:57 +0000 | [diff] [blame] | 305 |         n = len(population) | 
 | 306 |         if not 0 <= k <= n: | 
 | 307 |             raise ValueError, "sample larger than population" | 
| Raymond Hettinger | 8b9aa8d | 2003-01-04 05:20:33 +0000 | [diff] [blame] | 308 |         random = self.random | 
| Raymond Hettinger | fdbe522 | 2003-06-13 07:01:51 +0000 | [diff] [blame] | 309 |         _int = int | 
| Raymond Hettinger | c0b4034 | 2002-11-13 15:26:37 +0000 | [diff] [blame] | 310 |         result = [None] * k | 
| Raymond Hettinger | 91e27c2 | 2005-08-19 01:36:35 +0000 | [diff] [blame] | 311 |         setsize = 21        # size of a small set minus size of an empty list | 
 | 312 |         if k > 5: | 
| Tim Peters | 9e34c04 | 2005-08-26 15:20:46 +0000 | [diff] [blame] | 313 |             setsize += 4 ** _ceil(_log(k * 3, 4)) # table size for big sets | 
| Tim Peters | c17976e | 2006-04-01 00:26:53 +0000 | [diff] [blame] | 314 |         if n <= setsize or hasattr(population, "keys"): | 
 | 315 |             # An n-length list is smaller than a k-length set, or this is a | 
 | 316 |             # mapping type so the other algorithm wouldn't work. | 
| Raymond Hettinger | 311f419 | 2002-11-18 09:01:24 +0000 | [diff] [blame] | 317 |             pool = list(population) | 
 | 318 |             for i in xrange(k):         # invariant:  non-selected at [0,n-i) | 
| Raymond Hettinger | fdbe522 | 2003-06-13 07:01:51 +0000 | [diff] [blame] | 319 |                 j = _int(random() * (n-i)) | 
| Raymond Hettinger | 311f419 | 2002-11-18 09:01:24 +0000 | [diff] [blame] | 320 |                 result[i] = pool[j] | 
| Raymond Hettinger | 8b9aa8d | 2003-01-04 05:20:33 +0000 | [diff] [blame] | 321 |                 pool[j] = pool[n-i-1]   # move non-selected item into vacancy | 
| Raymond Hettinger | c0b4034 | 2002-11-13 15:26:37 +0000 | [diff] [blame] | 322 |         else: | 
| Raymond Hettinger | 66d09f1 | 2003-09-06 04:25:54 +0000 | [diff] [blame] | 323 |             try: | 
| Raymond Hettinger | 3c3346d | 2006-03-29 09:13:13 +0000 | [diff] [blame] | 324 |                 selected = set() | 
 | 325 |                 selected_add = selected.add | 
 | 326 |                 for i in xrange(k): | 
| Raymond Hettinger | fdbe522 | 2003-06-13 07:01:51 +0000 | [diff] [blame] | 327 |                     j = _int(random() * n) | 
| Raymond Hettinger | 3c3346d | 2006-03-29 09:13:13 +0000 | [diff] [blame] | 328 |                     while j in selected: | 
 | 329 |                         j = _int(random() * n) | 
 | 330 |                     selected_add(j) | 
 | 331 |                     result[i] = population[j] | 
| Tim Peters | c17976e | 2006-04-01 00:26:53 +0000 | [diff] [blame] | 332 |             except (TypeError, KeyError):   # handle (at least) sets | 
| Raymond Hettinger | 3c3346d | 2006-03-29 09:13:13 +0000 | [diff] [blame] | 333 |                 if isinstance(population, list): | 
 | 334 |                     raise | 
| Tim Peters | c17976e | 2006-04-01 00:26:53 +0000 | [diff] [blame] | 335 |                 return self.sample(tuple(population), k) | 
| Raymond Hettinger | 311f419 | 2002-11-18 09:01:24 +0000 | [diff] [blame] | 336 |         return result | 
| Raymond Hettinger | f24eb35 | 2002-11-12 17:41:57 +0000 | [diff] [blame] | 337 |  | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 338 | ## -------------------- real-valued distributions  ------------------- | 
 | 339 |  | 
 | 340 | ## -------------------- uniform distribution ------------------- | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 341 |  | 
 | 342 |     def uniform(self, a, b): | 
| Raymond Hettinger | 2c0cdca | 2009-06-11 23:14:53 +0000 | [diff] [blame] | 343 |         "Get a random number in the range [a, b) or [a, b] depending on rounding." | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 344 |         return a + (b-a) * self.random() | 
| Guido van Rossum | ff03b1a | 1994-03-09 12:55:02 +0000 | [diff] [blame] | 345 |  | 
| Raymond Hettinger | bbc50ea | 2008-03-23 13:32:32 +0000 | [diff] [blame] | 346 | ## -------------------- triangular -------------------- | 
 | 347 |  | 
| Raymond Hettinger | c4f7bab | 2008-03-23 19:37:53 +0000 | [diff] [blame] | 348 |     def triangular(self, low=0.0, high=1.0, mode=None): | 
| Raymond Hettinger | bbc50ea | 2008-03-23 13:32:32 +0000 | [diff] [blame] | 349 |         """Triangular distribution. | 
 | 350 |  | 
 | 351 |         Continuous distribution bounded by given lower and upper limits, | 
 | 352 |         and having a given mode value in-between. | 
 | 353 |  | 
 | 354 |         http://en.wikipedia.org/wiki/Triangular_distribution | 
 | 355 |  | 
 | 356 |         """ | 
 | 357 |         u = self.random() | 
| Raymond Hettinger | c4f7bab | 2008-03-23 19:37:53 +0000 | [diff] [blame] | 358 |         c = 0.5 if mode is None else (mode - low) / (high - low) | 
| Raymond Hettinger | bbc50ea | 2008-03-23 13:32:32 +0000 | [diff] [blame] | 359 |         if u > c: | 
| Raymond Hettinger | c4f7bab | 2008-03-23 19:37:53 +0000 | [diff] [blame] | 360 |             u = 1.0 - u | 
 | 361 |             c = 1.0 - c | 
| Raymond Hettinger | bbc50ea | 2008-03-23 13:32:32 +0000 | [diff] [blame] | 362 |             low, high = high, low | 
 | 363 |         return low + (high - low) * (u * c) ** 0.5 | 
 | 364 |  | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 365 | ## -------------------- normal distribution -------------------- | 
| Guido van Rossum | ff03b1a | 1994-03-09 12:55:02 +0000 | [diff] [blame] | 366 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 367 |     def normalvariate(self, mu, sigma): | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 368 |         """Normal distribution. | 
 | 369 |  | 
 | 370 |         mu is the mean, and sigma is the standard deviation. | 
| Raymond Hettinger | ef4d4bd | 2002-05-23 23:58:17 +0000 | [diff] [blame] | 371 |  | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 372 |         """ | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 373 |         # mu = mean, sigma = standard deviation | 
| Guido van Rossum | ff03b1a | 1994-03-09 12:55:02 +0000 | [diff] [blame] | 374 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 375 |         # Uses Kinderman and Monahan method. Reference: Kinderman, | 
 | 376 |         # A.J. and Monahan, J.F., "Computer generation of random | 
 | 377 |         # variables using the ratio of uniform deviates", ACM Trans | 
 | 378 |         # Math Software, 3, (1977), pp257-260. | 
| Guido van Rossum | ff03b1a | 1994-03-09 12:55:02 +0000 | [diff] [blame] | 379 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 380 |         random = self.random | 
| Raymond Hettinger | 42406e6 | 2005-04-30 09:02:51 +0000 | [diff] [blame] | 381 |         while 1: | 
| Tim Peters | 0c9886d | 2001-01-15 01:18:21 +0000 | [diff] [blame] | 382 |             u1 = random() | 
| Raymond Hettinger | 73ced7e | 2003-01-04 09:26:32 +0000 | [diff] [blame] | 383 |             u2 = 1.0 - random() | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 384 |             z = NV_MAGICCONST*(u1-0.5)/u2 | 
 | 385 |             zz = z*z/4.0 | 
 | 386 |             if zz <= -_log(u2): | 
 | 387 |                 break | 
 | 388 |         return mu + z*sigma | 
| Guido van Rossum | ff03b1a | 1994-03-09 12:55:02 +0000 | [diff] [blame] | 389 |  | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 390 | ## -------------------- lognormal distribution -------------------- | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 391 |  | 
 | 392 |     def lognormvariate(self, mu, sigma): | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 393 |         """Log normal distribution. | 
 | 394 |  | 
 | 395 |         If you take the natural logarithm of this distribution, you'll get a | 
 | 396 |         normal distribution with mean mu and standard deviation sigma. | 
 | 397 |         mu can have any value, and sigma must be greater than zero. | 
| Raymond Hettinger | ef4d4bd | 2002-05-23 23:58:17 +0000 | [diff] [blame] | 398 |  | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 399 |         """ | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 400 |         return _exp(self.normalvariate(mu, sigma)) | 
 | 401 |  | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 402 | ## -------------------- exponential distribution -------------------- | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 403 |  | 
 | 404 |     def expovariate(self, lambd): | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 405 |         """Exponential distribution. | 
 | 406 |  | 
| Mark Dickinson | e6dc531 | 2009-01-07 17:48:33 +0000 | [diff] [blame] | 407 |         lambd is 1.0 divided by the desired mean.  It should be | 
 | 408 |         nonzero.  (The parameter would be called "lambda", but that is | 
 | 409 |         a reserved word in Python.)  Returned values range from 0 to | 
 | 410 |         positive infinity if lambd is positive, and from negative | 
 | 411 |         infinity to 0 if lambd is negative. | 
| Raymond Hettinger | ef4d4bd | 2002-05-23 23:58:17 +0000 | [diff] [blame] | 412 |  | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 413 |         """ | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 414 |         # lambd: rate lambd = 1/mean | 
 | 415 |         # ('lambda' is a Python reserved word) | 
 | 416 |  | 
 | 417 |         random = self.random | 
| Tim Peters | 0c9886d | 2001-01-15 01:18:21 +0000 | [diff] [blame] | 418 |         u = random() | 
 | 419 |         while u <= 1e-7: | 
 | 420 |             u = random() | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 421 |         return -_log(u)/lambd | 
| Guido van Rossum | ff03b1a | 1994-03-09 12:55:02 +0000 | [diff] [blame] | 422 |  | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 423 | ## -------------------- von Mises distribution -------------------- | 
| Guido van Rossum | ff03b1a | 1994-03-09 12:55:02 +0000 | [diff] [blame] | 424 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 425 |     def vonmisesvariate(self, mu, kappa): | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 426 |         """Circular data distribution. | 
| Raymond Hettinger | ef4d4bd | 2002-05-23 23:58:17 +0000 | [diff] [blame] | 427 |  | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 428 |         mu is the mean angle, expressed in radians between 0 and 2*pi, and | 
 | 429 |         kappa is the concentration parameter, which must be greater than or | 
 | 430 |         equal to zero.  If kappa is equal to zero, this distribution reduces | 
 | 431 |         to a uniform random angle over the range 0 to 2*pi. | 
| Raymond Hettinger | ef4d4bd | 2002-05-23 23:58:17 +0000 | [diff] [blame] | 432 |  | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 433 |         """ | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 434 |         # mu:    mean angle (in radians between 0 and 2*pi) | 
 | 435 |         # kappa: concentration parameter kappa (>= 0) | 
 | 436 |         # if kappa = 0 generate uniform random angle | 
 | 437 |  | 
 | 438 |         # Based upon an algorithm published in: Fisher, N.I., | 
 | 439 |         # "Statistical Analysis of Circular Data", Cambridge | 
 | 440 |         # University Press, 1993. | 
 | 441 |  | 
 | 442 |         # Thanks to Magnus Kessler for a correction to the | 
 | 443 |         # implementation of step 4. | 
 | 444 |  | 
 | 445 |         random = self.random | 
 | 446 |         if kappa <= 1e-6: | 
 | 447 |             return TWOPI * random() | 
 | 448 |  | 
 | 449 |         a = 1.0 + _sqrt(1.0 + 4.0 * kappa * kappa) | 
 | 450 |         b = (a - _sqrt(2.0 * a))/(2.0 * kappa) | 
 | 451 |         r = (1.0 + b * b)/(2.0 * b) | 
| Guido van Rossum | ff03b1a | 1994-03-09 12:55:02 +0000 | [diff] [blame] | 452 |  | 
| Raymond Hettinger | 42406e6 | 2005-04-30 09:02:51 +0000 | [diff] [blame] | 453 |         while 1: | 
| Tim Peters | 0c9886d | 2001-01-15 01:18:21 +0000 | [diff] [blame] | 454 |             u1 = random() | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 455 |  | 
 | 456 |             z = _cos(_pi * u1) | 
 | 457 |             f = (1.0 + r * z)/(r + z) | 
 | 458 |             c = kappa * (r - f) | 
 | 459 |  | 
 | 460 |             u2 = random() | 
 | 461 |  | 
| Raymond Hettinger | 42406e6 | 2005-04-30 09:02:51 +0000 | [diff] [blame] | 462 |             if u2 < c * (2.0 - c) or u2 <= c * _exp(1.0 - c): | 
| Tim Peters | 0c9886d | 2001-01-15 01:18:21 +0000 | [diff] [blame] | 463 |                 break | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 464 |  | 
 | 465 |         u3 = random() | 
 | 466 |         if u3 > 0.5: | 
 | 467 |             theta = (mu % TWOPI) + _acos(f) | 
 | 468 |         else: | 
 | 469 |             theta = (mu % TWOPI) - _acos(f) | 
 | 470 |  | 
 | 471 |         return theta | 
 | 472 |  | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 473 | ## -------------------- gamma distribution -------------------- | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 474 |  | 
 | 475 |     def gammavariate(self, alpha, beta): | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 476 |         """Gamma distribution.  Not the gamma function! | 
 | 477 |  | 
 | 478 |         Conditions on the parameters are alpha > 0 and beta > 0. | 
 | 479 |  | 
 | 480 |         """ | 
| Tim Peters | 8ac1495 | 2002-05-23 15:15:30 +0000 | [diff] [blame] | 481 |  | 
| Raymond Hettinger | b760efb | 2002-05-14 06:40:34 +0000 | [diff] [blame] | 482 |         # alpha > 0, beta > 0, mean is alpha*beta, variance is alpha*beta**2 | 
| Tim Peters | 8ac1495 | 2002-05-23 15:15:30 +0000 | [diff] [blame] | 483 |  | 
| Guido van Rossum | 570764d | 2002-05-14 14:08:12 +0000 | [diff] [blame] | 484 |         # Warning: a few older sources define the gamma distribution in terms | 
 | 485 |         # of alpha > -1.0 | 
 | 486 |         if alpha <= 0.0 or beta <= 0.0: | 
 | 487 |             raise ValueError, 'gammavariate: alpha and beta must be > 0.0' | 
| Tim Peters | 8ac1495 | 2002-05-23 15:15:30 +0000 | [diff] [blame] | 488 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 489 |         random = self.random | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 490 |         if alpha > 1.0: | 
 | 491 |  | 
 | 492 |             # Uses R.C.H. Cheng, "The generation of Gamma | 
 | 493 |             # variables with non-integral shape parameters", | 
 | 494 |             # Applied Statistics, (1977), 26, No. 1, p71-74 | 
 | 495 |  | 
| Raymond Hettinger | ca6cdc2 | 2002-05-13 23:40:14 +0000 | [diff] [blame] | 496 |             ainv = _sqrt(2.0 * alpha - 1.0) | 
 | 497 |             bbb = alpha - LOG4 | 
 | 498 |             ccc = alpha + ainv | 
| Tim Peters | 8ac1495 | 2002-05-23 15:15:30 +0000 | [diff] [blame] | 499 |  | 
| Raymond Hettinger | 42406e6 | 2005-04-30 09:02:51 +0000 | [diff] [blame] | 500 |             while 1: | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 501 |                 u1 = random() | 
| Raymond Hettinger | 73ced7e | 2003-01-04 09:26:32 +0000 | [diff] [blame] | 502 |                 if not 1e-7 < u1 < .9999999: | 
 | 503 |                     continue | 
 | 504 |                 u2 = 1.0 - random() | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 505 |                 v = _log(u1/(1.0-u1))/ainv | 
 | 506 |                 x = alpha*_exp(v) | 
 | 507 |                 z = u1*u1*u2 | 
 | 508 |                 r = bbb+ccc*v-x | 
 | 509 |                 if r + SG_MAGICCONST - 4.5*z >= 0.0 or r >= _log(z): | 
| Raymond Hettinger | b760efb | 2002-05-14 06:40:34 +0000 | [diff] [blame] | 510 |                     return x * beta | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 511 |  | 
 | 512 |         elif alpha == 1.0: | 
 | 513 |             # expovariate(1) | 
 | 514 |             u = random() | 
 | 515 |             while u <= 1e-7: | 
 | 516 |                 u = random() | 
| Raymond Hettinger | b760efb | 2002-05-14 06:40:34 +0000 | [diff] [blame] | 517 |             return -_log(u) * beta | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 518 |  | 
 | 519 |         else:   # alpha is between 0 and 1 (exclusive) | 
 | 520 |  | 
 | 521 |             # Uses ALGORITHM GS of Statistical Computing - Kennedy & Gentle | 
 | 522 |  | 
| Raymond Hettinger | 42406e6 | 2005-04-30 09:02:51 +0000 | [diff] [blame] | 523 |             while 1: | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 524 |                 u = random() | 
 | 525 |                 b = (_e + alpha)/_e | 
 | 526 |                 p = b*u | 
 | 527 |                 if p <= 1.0: | 
| Raymond Hettinger | 42406e6 | 2005-04-30 09:02:51 +0000 | [diff] [blame] | 528 |                     x = p ** (1.0/alpha) | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 529 |                 else: | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 530 |                     x = -_log((b-p)/alpha) | 
 | 531 |                 u1 = random() | 
| Raymond Hettinger | 42406e6 | 2005-04-30 09:02:51 +0000 | [diff] [blame] | 532 |                 if p > 1.0: | 
 | 533 |                     if u1 <= x ** (alpha - 1.0): | 
 | 534 |                         break | 
 | 535 |                 elif u1 <= _exp(-x): | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 536 |                     break | 
| Raymond Hettinger | b760efb | 2002-05-14 06:40:34 +0000 | [diff] [blame] | 537 |             return x * beta | 
 | 538 |  | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 539 | ## -------------------- Gauss (faster alternative) -------------------- | 
| Guido van Rossum | 95bfcda | 1994-03-09 14:21:05 +0000 | [diff] [blame] | 540 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 541 |     def gauss(self, mu, sigma): | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 542 |         """Gaussian distribution. | 
 | 543 |  | 
 | 544 |         mu is the mean, and sigma is the standard deviation.  This is | 
 | 545 |         slightly faster than the normalvariate() function. | 
 | 546 |  | 
 | 547 |         Not thread-safe without a lock around calls. | 
| Raymond Hettinger | ef4d4bd | 2002-05-23 23:58:17 +0000 | [diff] [blame] | 548 |  | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 549 |         """ | 
| Guido van Rossum | cc32ac9 | 1994-03-15 16:10:24 +0000 | [diff] [blame] | 550 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 551 |         # When x and y are two variables from [0, 1), uniformly | 
 | 552 |         # distributed, then | 
 | 553 |         # | 
 | 554 |         #    cos(2*pi*x)*sqrt(-2*log(1-y)) | 
 | 555 |         #    sin(2*pi*x)*sqrt(-2*log(1-y)) | 
 | 556 |         # | 
 | 557 |         # are two *independent* variables with normal distribution | 
 | 558 |         # (mu = 0, sigma = 1). | 
 | 559 |         # (Lambert Meertens) | 
 | 560 |         # (corrected version; bug discovered by Mike Miller, fixed by LM) | 
| Guido van Rossum | cc32ac9 | 1994-03-15 16:10:24 +0000 | [diff] [blame] | 561 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 562 |         # Multithreading note: When two threads call this function | 
 | 563 |         # simultaneously, it is possible that they will receive the | 
 | 564 |         # same return value.  The window is very small though.  To | 
 | 565 |         # avoid this, you have to use a lock around all calls.  (I | 
 | 566 |         # didn't want to slow this down in the serial case by using a | 
 | 567 |         # lock here.) | 
| Guido van Rossum | d03e119 | 1998-05-29 17:51:31 +0000 | [diff] [blame] | 568 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 569 |         random = self.random | 
 | 570 |         z = self.gauss_next | 
 | 571 |         self.gauss_next = None | 
 | 572 |         if z is None: | 
 | 573 |             x2pi = random() * TWOPI | 
 | 574 |             g2rad = _sqrt(-2.0 * _log(1.0 - random())) | 
 | 575 |             z = _cos(x2pi) * g2rad | 
 | 576 |             self.gauss_next = _sin(x2pi) * g2rad | 
| Guido van Rossum | cc32ac9 | 1994-03-15 16:10:24 +0000 | [diff] [blame] | 577 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 578 |         return mu + z*sigma | 
| Guido van Rossum | 95bfcda | 1994-03-09 14:21:05 +0000 | [diff] [blame] | 579 |  | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 580 | ## -------------------- beta -------------------- | 
| Tim Peters | 85e2e47 | 2001-01-26 06:49:56 +0000 | [diff] [blame] | 581 | ## See | 
 | 582 | ## http://sourceforge.net/bugs/?func=detailbug&bug_id=130030&group_id=5470 | 
 | 583 | ## for Ivan Frohne's insightful analysis of why the original implementation: | 
 | 584 | ## | 
 | 585 | ##    def betavariate(self, alpha, beta): | 
 | 586 | ##        # Discrete Event Simulation in C, pp 87-88. | 
 | 587 | ## | 
 | 588 | ##        y = self.expovariate(alpha) | 
 | 589 | ##        z = self.expovariate(1.0/beta) | 
 | 590 | ##        return z/(y+z) | 
 | 591 | ## | 
 | 592 | ## was dead wrong, and how it probably got that way. | 
| Guido van Rossum | 95bfcda | 1994-03-09 14:21:05 +0000 | [diff] [blame] | 593 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 594 |     def betavariate(self, alpha, beta): | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 595 |         """Beta distribution. | 
 | 596 |  | 
| Raymond Hettinger | 1b0ce85 | 2007-01-19 18:07:18 +0000 | [diff] [blame] | 597 |         Conditions on the parameters are alpha > 0 and beta > 0. | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 598 |         Returned values range between 0 and 1. | 
| Raymond Hettinger | ef4d4bd | 2002-05-23 23:58:17 +0000 | [diff] [blame] | 599 |  | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 600 |         """ | 
| Raymond Hettinger | ef4d4bd | 2002-05-23 23:58:17 +0000 | [diff] [blame] | 601 |  | 
| Tim Peters | 85e2e47 | 2001-01-26 06:49:56 +0000 | [diff] [blame] | 602 |         # This version due to Janne Sinkkonen, and matches all the std | 
 | 603 |         # texts (e.g., Knuth Vol 2 Ed 3 pg 134 "the beta distribution"). | 
 | 604 |         y = self.gammavariate(alpha, 1.) | 
 | 605 |         if y == 0: | 
 | 606 |             return 0.0 | 
 | 607 |         else: | 
 | 608 |             return y / (y + self.gammavariate(beta, 1.)) | 
| Guido van Rossum | 95bfcda | 1994-03-09 14:21:05 +0000 | [diff] [blame] | 609 |  | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 610 | ## -------------------- Pareto -------------------- | 
| Guido van Rossum | cf4559a | 1997-12-02 02:47:39 +0000 | [diff] [blame] | 611 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 612 |     def paretovariate(self, alpha): | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 613 |         """Pareto distribution.  alpha is the shape parameter.""" | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 614 |         # Jain, pg. 495 | 
| Guido van Rossum | cf4559a | 1997-12-02 02:47:39 +0000 | [diff] [blame] | 615 |  | 
| Raymond Hettinger | 73ced7e | 2003-01-04 09:26:32 +0000 | [diff] [blame] | 616 |         u = 1.0 - self.random() | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 617 |         return 1.0 / pow(u, 1.0/alpha) | 
| Guido van Rossum | cf4559a | 1997-12-02 02:47:39 +0000 | [diff] [blame] | 618 |  | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 619 | ## -------------------- Weibull -------------------- | 
| Guido van Rossum | cf4559a | 1997-12-02 02:47:39 +0000 | [diff] [blame] | 620 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 621 |     def weibullvariate(self, alpha, beta): | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 622 |         """Weibull distribution. | 
 | 623 |  | 
 | 624 |         alpha is the scale parameter and beta is the shape parameter. | 
| Raymond Hettinger | ef4d4bd | 2002-05-23 23:58:17 +0000 | [diff] [blame] | 625 |  | 
| Raymond Hettinger | c32f033 | 2002-05-23 19:44:49 +0000 | [diff] [blame] | 626 |         """ | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 627 |         # Jain, pg. 499; bug fix courtesy Bill Arms | 
| Guido van Rossum | cf4559a | 1997-12-02 02:47:39 +0000 | [diff] [blame] | 628 |  | 
| Raymond Hettinger | 73ced7e | 2003-01-04 09:26:32 +0000 | [diff] [blame] | 629 |         u = 1.0 - self.random() | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 630 |         return alpha * pow(-_log(u), 1.0/beta) | 
| Guido van Rossum | 6c395ba | 1999-08-18 13:53:28 +0000 | [diff] [blame] | 631 |  | 
| Raymond Hettinger | 40f6217 | 2002-12-29 23:03:38 +0000 | [diff] [blame] | 632 | ## -------------------- Wichmann-Hill ------------------- | 
 | 633 |  | 
 | 634 | class WichmannHill(Random): | 
 | 635 |  | 
 | 636 |     VERSION = 1     # used by getstate/setstate | 
 | 637 |  | 
 | 638 |     def seed(self, a=None): | 
 | 639 |         """Initialize internal state from hashable object. | 
 | 640 |  | 
| Raymond Hettinger | 23f1241 | 2004-09-13 22:23:21 +0000 | [diff] [blame] | 641 |         None or no argument seeds from current time or from an operating | 
 | 642 |         system specific randomness source if available. | 
| Raymond Hettinger | 40f6217 | 2002-12-29 23:03:38 +0000 | [diff] [blame] | 643 |  | 
 | 644 |         If a is not None or an int or long, hash(a) is used instead. | 
 | 645 |  | 
 | 646 |         If a is an int or long, a is used directly.  Distinct values between | 
 | 647 |         0 and 27814431486575L inclusive are guaranteed to yield distinct | 
 | 648 |         internal states (this guarantee is specific to the default | 
 | 649 |         Wichmann-Hill generator). | 
 | 650 |         """ | 
 | 651 |  | 
 | 652 |         if a is None: | 
| Raymond Hettinger | c1c43ca | 2004-09-05 00:00:42 +0000 | [diff] [blame] | 653 |             try: | 
 | 654 |                 a = long(_hexlify(_urandom(16)), 16) | 
 | 655 |             except NotImplementedError: | 
| Raymond Hettinger | 356a459 | 2004-08-30 06:14:31 +0000 | [diff] [blame] | 656 |                 import time | 
 | 657 |                 a = long(time.time() * 256) # use fractional seconds | 
| Raymond Hettinger | 40f6217 | 2002-12-29 23:03:38 +0000 | [diff] [blame] | 658 |  | 
 | 659 |         if not isinstance(a, (int, long)): | 
 | 660 |             a = hash(a) | 
 | 661 |  | 
 | 662 |         a, x = divmod(a, 30268) | 
 | 663 |         a, y = divmod(a, 30306) | 
 | 664 |         a, z = divmod(a, 30322) | 
 | 665 |         self._seed = int(x)+1, int(y)+1, int(z)+1 | 
 | 666 |  | 
 | 667 |         self.gauss_next = None | 
 | 668 |  | 
 | 669 |     def random(self): | 
 | 670 |         """Get the next random number in the range [0.0, 1.0).""" | 
 | 671 |  | 
 | 672 |         # Wichman-Hill random number generator. | 
 | 673 |         # | 
 | 674 |         # Wichmann, B. A. & Hill, I. D. (1982) | 
 | 675 |         # Algorithm AS 183: | 
 | 676 |         # An efficient and portable pseudo-random number generator | 
 | 677 |         # Applied Statistics 31 (1982) 188-190 | 
 | 678 |         # | 
 | 679 |         # see also: | 
 | 680 |         #        Correction to Algorithm AS 183 | 
 | 681 |         #        Applied Statistics 33 (1984) 123 | 
 | 682 |         # | 
 | 683 |         #        McLeod, A. I. (1985) | 
 | 684 |         #        A remark on Algorithm AS 183 | 
 | 685 |         #        Applied Statistics 34 (1985),198-200 | 
 | 686 |  | 
 | 687 |         # This part is thread-unsafe: | 
 | 688 |         # BEGIN CRITICAL SECTION | 
 | 689 |         x, y, z = self._seed | 
 | 690 |         x = (171 * x) % 30269 | 
 | 691 |         y = (172 * y) % 30307 | 
 | 692 |         z = (170 * z) % 30323 | 
 | 693 |         self._seed = x, y, z | 
 | 694 |         # END CRITICAL SECTION | 
 | 695 |  | 
 | 696 |         # Note:  on a platform using IEEE-754 double arithmetic, this can | 
 | 697 |         # never return 0.0 (asserted by Tim; proof too long for a comment). | 
 | 698 |         return (x/30269.0 + y/30307.0 + z/30323.0) % 1.0 | 
 | 699 |  | 
 | 700 |     def getstate(self): | 
 | 701 |         """Return internal state; can be passed to setstate() later.""" | 
 | 702 |         return self.VERSION, self._seed, self.gauss_next | 
 | 703 |  | 
 | 704 |     def setstate(self, state): | 
 | 705 |         """Restore internal state from object returned by getstate().""" | 
 | 706 |         version = state[0] | 
 | 707 |         if version == 1: | 
 | 708 |             version, self._seed, self.gauss_next = state | 
 | 709 |         else: | 
 | 710 |             raise ValueError("state with version %s passed to " | 
 | 711 |                              "Random.setstate() of version %s" % | 
 | 712 |                              (version, self.VERSION)) | 
 | 713 |  | 
 | 714 |     def jumpahead(self, n): | 
 | 715 |         """Act as if n calls to random() were made, but quickly. | 
 | 716 |  | 
 | 717 |         n is an int, greater than or equal to 0. | 
 | 718 |  | 
 | 719 |         Example use:  If you have 2 threads and know that each will | 
 | 720 |         consume no more than a million random numbers, create two Random | 
 | 721 |         objects r1 and r2, then do | 
 | 722 |             r2.setstate(r1.getstate()) | 
 | 723 |             r2.jumpahead(1000000) | 
 | 724 |         Then r1 and r2 will use guaranteed-disjoint segments of the full | 
 | 725 |         period. | 
 | 726 |         """ | 
 | 727 |  | 
 | 728 |         if not n >= 0: | 
 | 729 |             raise ValueError("n must be >= 0") | 
 | 730 |         x, y, z = self._seed | 
 | 731 |         x = int(x * pow(171, n, 30269)) % 30269 | 
 | 732 |         y = int(y * pow(172, n, 30307)) % 30307 | 
 | 733 |         z = int(z * pow(170, n, 30323)) % 30323 | 
 | 734 |         self._seed = x, y, z | 
 | 735 |  | 
 | 736 |     def __whseed(self, x=0, y=0, z=0): | 
 | 737 |         """Set the Wichmann-Hill seed from (x, y, z). | 
 | 738 |  | 
 | 739 |         These must be integers in the range [0, 256). | 
 | 740 |         """ | 
 | 741 |  | 
 | 742 |         if not type(x) == type(y) == type(z) == int: | 
 | 743 |             raise TypeError('seeds must be integers') | 
 | 744 |         if not (0 <= x < 256 and 0 <= y < 256 and 0 <= z < 256): | 
 | 745 |             raise ValueError('seeds must be in range(0, 256)') | 
 | 746 |         if 0 == x == y == z: | 
 | 747 |             # Initialize from current time | 
 | 748 |             import time | 
 | 749 |             t = long(time.time() * 256) | 
 | 750 |             t = int((t&0xffffff) ^ (t>>24)) | 
 | 751 |             t, x = divmod(t, 256) | 
 | 752 |             t, y = divmod(t, 256) | 
 | 753 |             t, z = divmod(t, 256) | 
 | 754 |         # Zero is a poor seed, so substitute 1 | 
 | 755 |         self._seed = (x or 1, y or 1, z or 1) | 
 | 756 |  | 
 | 757 |         self.gauss_next = None | 
 | 758 |  | 
 | 759 |     def whseed(self, a=None): | 
 | 760 |         """Seed from hashable object's hash code. | 
 | 761 |  | 
 | 762 |         None or no argument seeds from current time.  It is not guaranteed | 
 | 763 |         that objects with distinct hash codes lead to distinct internal | 
 | 764 |         states. | 
 | 765 |  | 
 | 766 |         This is obsolete, provided for compatibility with the seed routine | 
 | 767 |         used prior to Python 2.1.  Use the .seed() method instead. | 
 | 768 |         """ | 
 | 769 |  | 
 | 770 |         if a is None: | 
 | 771 |             self.__whseed() | 
 | 772 |             return | 
 | 773 |         a = hash(a) | 
 | 774 |         a, x = divmod(a, 256) | 
 | 775 |         a, y = divmod(a, 256) | 
 | 776 |         a, z = divmod(a, 256) | 
 | 777 |         x = (x + a) % 256 or 1 | 
 | 778 |         y = (y + a) % 256 or 1 | 
 | 779 |         z = (z + a) % 256 or 1 | 
 | 780 |         self.__whseed(x, y, z) | 
 | 781 |  | 
| Raymond Hettinger | 23f1241 | 2004-09-13 22:23:21 +0000 | [diff] [blame] | 782 | ## --------------- Operating System Random Source  ------------------ | 
| Raymond Hettinger | 356a459 | 2004-08-30 06:14:31 +0000 | [diff] [blame] | 783 |  | 
| Raymond Hettinger | 23f1241 | 2004-09-13 22:23:21 +0000 | [diff] [blame] | 784 | class SystemRandom(Random): | 
 | 785 |     """Alternate random number generator using sources provided | 
 | 786 |     by the operating system (such as /dev/urandom on Unix or | 
 | 787 |     CryptGenRandom on Windows). | 
| Raymond Hettinger | 356a459 | 2004-08-30 06:14:31 +0000 | [diff] [blame] | 788 |  | 
 | 789 |      Not available on all systems (see os.urandom() for details). | 
 | 790 |     """ | 
 | 791 |  | 
 | 792 |     def random(self): | 
 | 793 |         """Get the next random number in the range [0.0, 1.0).""" | 
| Tim Peters | 7c2a85b | 2004-08-31 02:19:55 +0000 | [diff] [blame] | 794 |         return (long(_hexlify(_urandom(7)), 16) >> 3) * RECIP_BPF | 
| Raymond Hettinger | 356a459 | 2004-08-30 06:14:31 +0000 | [diff] [blame] | 795 |  | 
 | 796 |     def getrandbits(self, k): | 
 | 797 |         """getrandbits(k) -> x.  Generates a long int with k random bits.""" | 
| Raymond Hettinger | 356a459 | 2004-08-30 06:14:31 +0000 | [diff] [blame] | 798 |         if k <= 0: | 
 | 799 |             raise ValueError('number of bits must be greater than zero') | 
 | 800 |         if k != int(k): | 
 | 801 |             raise TypeError('number of bits should be an integer') | 
 | 802 |         bytes = (k + 7) // 8                    # bits / 8 and rounded up | 
 | 803 |         x = long(_hexlify(_urandom(bytes)), 16) | 
 | 804 |         return x >> (bytes * 8 - k)             # trim excess bits | 
 | 805 |  | 
 | 806 |     def _stub(self, *args, **kwds): | 
| Raymond Hettinger | 23f1241 | 2004-09-13 22:23:21 +0000 | [diff] [blame] | 807 |         "Stub method.  Not used for a system random number generator." | 
| Raymond Hettinger | 356a459 | 2004-08-30 06:14:31 +0000 | [diff] [blame] | 808 |         return None | 
 | 809 |     seed = jumpahead = _stub | 
 | 810 |  | 
 | 811 |     def _notimplemented(self, *args, **kwds): | 
| Raymond Hettinger | 23f1241 | 2004-09-13 22:23:21 +0000 | [diff] [blame] | 812 |         "Method should not be called for a system random number generator." | 
 | 813 |         raise NotImplementedError('System entropy source does not have state.') | 
| Raymond Hettinger | 356a459 | 2004-08-30 06:14:31 +0000 | [diff] [blame] | 814 |     getstate = setstate = _notimplemented | 
 | 815 |  | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 816 | ## -------------------- test program -------------------- | 
| Guido van Rossum | ff03b1a | 1994-03-09 12:55:02 +0000 | [diff] [blame] | 817 |  | 
| Raymond Hettinger | 6229713 | 2003-08-30 01:24:19 +0000 | [diff] [blame] | 818 | def _test_generator(n, func, args): | 
| Tim Peters | 0c9886d | 2001-01-15 01:18:21 +0000 | [diff] [blame] | 819 |     import time | 
| Raymond Hettinger | 6229713 | 2003-08-30 01:24:19 +0000 | [diff] [blame] | 820 |     print n, 'times', func.__name__ | 
| Raymond Hettinger | b98154e | 2003-05-24 17:26:02 +0000 | [diff] [blame] | 821 |     total = 0.0 | 
| Tim Peters | 0c9886d | 2001-01-15 01:18:21 +0000 | [diff] [blame] | 822 |     sqsum = 0.0 | 
 | 823 |     smallest = 1e10 | 
 | 824 |     largest = -1e10 | 
 | 825 |     t0 = time.time() | 
 | 826 |     for i in range(n): | 
| Raymond Hettinger | 6229713 | 2003-08-30 01:24:19 +0000 | [diff] [blame] | 827 |         x = func(*args) | 
| Raymond Hettinger | b98154e | 2003-05-24 17:26:02 +0000 | [diff] [blame] | 828 |         total += x | 
| Tim Peters | 0c9886d | 2001-01-15 01:18:21 +0000 | [diff] [blame] | 829 |         sqsum = sqsum + x*x | 
 | 830 |         smallest = min(x, smallest) | 
 | 831 |         largest = max(x, largest) | 
 | 832 |     t1 = time.time() | 
 | 833 |     print round(t1-t0, 3), 'sec,', | 
| Raymond Hettinger | b98154e | 2003-05-24 17:26:02 +0000 | [diff] [blame] | 834 |     avg = total/n | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 835 |     stddev = _sqrt(sqsum/n - avg*avg) | 
| Tim Peters | 0c9886d | 2001-01-15 01:18:21 +0000 | [diff] [blame] | 836 |     print 'avg %g, stddev %g, min %g, max %g' % \ | 
 | 837 |               (avg, stddev, smallest, largest) | 
| Guido van Rossum | ff03b1a | 1994-03-09 12:55:02 +0000 | [diff] [blame] | 838 |  | 
| Raymond Hettinger | f24eb35 | 2002-11-12 17:41:57 +0000 | [diff] [blame] | 839 |  | 
 | 840 | def _test(N=2000): | 
| Raymond Hettinger | 6229713 | 2003-08-30 01:24:19 +0000 | [diff] [blame] | 841 |     _test_generator(N, random, ()) | 
 | 842 |     _test_generator(N, normalvariate, (0.0, 1.0)) | 
 | 843 |     _test_generator(N, lognormvariate, (0.0, 1.0)) | 
 | 844 |     _test_generator(N, vonmisesvariate, (0.0, 1.0)) | 
 | 845 |     _test_generator(N, gammavariate, (0.01, 1.0)) | 
 | 846 |     _test_generator(N, gammavariate, (0.1, 1.0)) | 
 | 847 |     _test_generator(N, gammavariate, (0.1, 2.0)) | 
 | 848 |     _test_generator(N, gammavariate, (0.5, 1.0)) | 
 | 849 |     _test_generator(N, gammavariate, (0.9, 1.0)) | 
 | 850 |     _test_generator(N, gammavariate, (1.0, 1.0)) | 
 | 851 |     _test_generator(N, gammavariate, (2.0, 1.0)) | 
 | 852 |     _test_generator(N, gammavariate, (20.0, 1.0)) | 
 | 853 |     _test_generator(N, gammavariate, (200.0, 1.0)) | 
 | 854 |     _test_generator(N, gauss, (0.0, 1.0)) | 
 | 855 |     _test_generator(N, betavariate, (3.0, 3.0)) | 
| Raymond Hettinger | bbc50ea | 2008-03-23 13:32:32 +0000 | [diff] [blame] | 856 |     _test_generator(N, triangular, (0.0, 1.0, 1.0/3.0)) | 
| Tim Peters | cd80410 | 2001-01-25 20:25:57 +0000 | [diff] [blame] | 857 |  | 
| Tim Peters | 715c4c4 | 2001-01-26 22:56:56 +0000 | [diff] [blame] | 858 | # Create one instance, seeded from current time, and export its methods | 
| Raymond Hettinger | 40f6217 | 2002-12-29 23:03:38 +0000 | [diff] [blame] | 859 | # as module-level functions.  The functions share state across all uses | 
 | 860 | #(both in the user's code and in the Python libraries), but that's fine | 
 | 861 | # for most programs and is easier for the casual user than making them | 
 | 862 | # instantiate their own Random() instance. | 
 | 863 |  | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 864 | _inst = Random() | 
 | 865 | seed = _inst.seed | 
 | 866 | random = _inst.random | 
 | 867 | uniform = _inst.uniform | 
| Raymond Hettinger | bbc50ea | 2008-03-23 13:32:32 +0000 | [diff] [blame] | 868 | triangular = _inst.triangular | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 869 | randint = _inst.randint | 
 | 870 | choice = _inst.choice | 
 | 871 | randrange = _inst.randrange | 
| Raymond Hettinger | f24eb35 | 2002-11-12 17:41:57 +0000 | [diff] [blame] | 872 | sample = _inst.sample | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 873 | shuffle = _inst.shuffle | 
 | 874 | normalvariate = _inst.normalvariate | 
 | 875 | lognormvariate = _inst.lognormvariate | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 876 | expovariate = _inst.expovariate | 
 | 877 | vonmisesvariate = _inst.vonmisesvariate | 
 | 878 | gammavariate = _inst.gammavariate | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 879 | gauss = _inst.gauss | 
 | 880 | betavariate = _inst.betavariate | 
 | 881 | paretovariate = _inst.paretovariate | 
 | 882 | weibullvariate = _inst.weibullvariate | 
 | 883 | getstate = _inst.getstate | 
 | 884 | setstate = _inst.setstate | 
| Tim Peters | d52269b | 2001-01-25 06:23:18 +0000 | [diff] [blame] | 885 | jumpahead = _inst.jumpahead | 
| Raymond Hettinger | 2f726e9 | 2003-10-05 09:09:15 +0000 | [diff] [blame] | 886 | getrandbits = _inst.getrandbits | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 887 |  | 
| Guido van Rossum | ff03b1a | 1994-03-09 12:55:02 +0000 | [diff] [blame] | 888 | if __name__ == '__main__': | 
| Tim Peters | d7b5e88 | 2001-01-25 03:36:26 +0000 | [diff] [blame] | 889 |     _test() |