-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathjavarandom.py
171 lines (131 loc) · 4.68 KB
/
javarandom.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
import time
import math
class Random(object):
"""
An implementation of the Java SE 6 random number generator.
Java's RNG is based on a classic Knuth-style linear congruential formula,
as described in
http://download.oracle.com/javase/6/docs/api/java/util/Random.html. This
makes it quite trivial to reimplement and port to other platforms.
This class should be bit-for-bit compatible with any Java RNG.
This class is not thread-safe. For deterministic behavior, lock or
synchronize all accesses to this class per-instance.
"""
def __init__(self, seed = None):
"""
Create a new random number generator.
"""
if seed is None:
seed = int(time.time() * 1000)
self.seed = seed
self.nextNextGaussian = None
def setSeed(self, seed):
"""
Explicit setter for seed, for compatibility with Java.
"""
self.seed = seed
@property
def seed(self):
return self._seed
@seed.setter
def seed(self, seed):
self._seed = (seed ^ 0x5deece66d) & ((1 << 48) - 1)
def next(self, bits):
"""
Generate the next random number.
As in Java, the general rule is that this method returns an int that
is `bits` bits long, where each bit is nearly equally likely to be 0
or 1.
"""
if bits < 1:
bits = 1
elif bits > 32:
bits = 32
self._seed = (self._seed * 0x5deece66d + 0xb) & ((1 << 48) - 1)
retval = self._seed >> (48 - bits)
# Python and Java don't really agree on how ints work. This converts
# the unsigned generated int into a signed int if necessary.
if retval & (1 << 31):
retval -= (1 << 32)
return retval
def nextBytes(self, l):
"""
Replace every item in `l` with a random byte.
"""
for i in range(0, len(l)):
if not i % 4:
n = self.nextInt()
b = n & 0xff
# Flip signs. Ugh.
if b & 0x80:
b -= 0x100
l[i] = b
n >>= 8
def nextInt(self, n = None):
"""
Return a random int in [0, `n`).
If `n` is not supplied, a random 32-bit integer will be returned.
"""
if n is None:
return self.next(32)
if n <= 0:
raise ValueError("Argument must be positive!")
# This tricky chunk of code comes straight from the Java spec. In
# essence, the algorithm tends to have much better entropy in the
# higher bits of the seed, so this little bundle of joy is used to try
# to reject values which would be obviously biased. We do have an easy
# out for power-of-two n, in which case we can call next directly.
# Is this a power of two?
if not (n & (n - 1)):
return (n * self.next(31)) >> 31
bits = self.next(31)
val = bits % n
while (bits - val + n - 1) < 0:
bits = self.next(31)
val = bits % n
return val
def nextLong(self):
"""
Return a random long.
Java longs are 64 bits wide, but the generator is only 48 bits wide,
so we generate two 32-bit numbers and glue them together.
"""
return (self.next(32) << 32) + self.next(32)
def nextBoolean(self):
"""
Return a random bool.
"""
return bool(self.next(1))
def nextFloat(self):
"""
Return a random float in (0, 1).
Python floats always carry double precision, so this function's return
values may appear less-than-random, but they are random in single
precision space.
"""
return self.next(24) / float(1 << 24)
def nextDouble(self):
"""
Return a random float in (0, 1).
"""
return ((self.next(26) << 27) + self.next(27)) / float(1 << 53)
def nextGaussian(self):
"""
Return a normally-distributed double with mean 0 and standard
deviation 1.
This method may not be strict enough to perfectly match the produced
values of Java's Random.nextGaussian().
"""
if self.nextNextGaussian is None:
s = 0
while s == 0 or s >= 1:
v1 = 2 * self.nextDouble() - 1
v2 = 2 * self.nextDouble() - 1
s = v1 * v1 + v2 * v2
multiplier = math.sqrt(-2 * math.log(s) / s)
self.nextNextGaussian = v2 * multiplier
return v1 * multiplier
else:
retval = self.nextNextGaussian
self.nextNextGaussian = None
return retval