133 lines
4.9 KiB
Python
133 lines
4.9 KiB
Python
"""RandomNumGen module: contains the RandomNumGen class"""
|
|
|
|
__all__ = ['randHash', 'RandomNumGen']
|
|
|
|
from direct.directnotify import DirectNotifyGlobal
|
|
from panda3d.core import Mersenne
|
|
|
|
def randHash(num):
|
|
""" this returns a random 16-bit integer, given a seed integer.
|
|
It will always return the same output given the same input.
|
|
This is useful for repeatably mapping numbers with predictable
|
|
bit patterns (i.e. doIds or zoneIds) to numbers with random bit patterns
|
|
"""
|
|
rng = RandomNumGen(num)
|
|
return rng.randint(0, (1<<16) - 1)
|
|
|
|
class RandomNumGen:
|
|
notify = \
|
|
DirectNotifyGlobal.directNotify.newCategory("RandomNumGen")
|
|
|
|
def __init__(self, seed):
|
|
"""seed must be an integer or another RandomNumGen"""
|
|
if isinstance(seed, RandomNumGen):
|
|
# seed this rng with the other rng
|
|
rng = seed
|
|
seed = rng.randint(0, 1 << 16)
|
|
|
|
self.notify.debug("seed: " + str(seed))
|
|
seed = int(seed)
|
|
rng = Mersenne(seed)
|
|
self.__rng = rng
|
|
|
|
def __rand(self, N):
|
|
"""returns integer in [0..N)"""
|
|
"""
|
|
# using modulus biases the numbers a little bit
|
|
# the bias is worse for larger values of N
|
|
return self.__rng.getUint31() % N
|
|
"""
|
|
|
|
# this technique produces an even distribution.
|
|
# random.py would solve this problem like so:
|
|
# where:
|
|
# M=randomly generated number
|
|
# O=1 greater than the maximum value of M
|
|
# return int(float(M)*(float(N)/float(O))) # M*(N/O)
|
|
#
|
|
# that generally works fine, except that it relies
|
|
# on floating-point numbers, which are not guaranteed
|
|
# to produce identical results on different machines.
|
|
#
|
|
# for our purposes, we need an entirely-integer approach,
|
|
# since integer operations *are* guaranteed to produce
|
|
# identical results on different machines.
|
|
#
|
|
# SO, we take the equation M*(N/O) and change the order of
|
|
# operations to (M*N)/O.
|
|
#
|
|
# this requires that we have the ability to hold the result of
|
|
# M*N. Luckily, Python has support for large integers. One
|
|
# alternative would be to limit the RNG to a 16-bit range,
|
|
# in which case we could do this math down in C++; but 16 bits
|
|
# really doesn't provide a large enough range (0..65535).
|
|
# Finally, since our O happens to be a power of two (0x80000000),
|
|
# we can replace the divide with a shift.
|
|
# boo-ya
|
|
|
|
# the maximum for N ought to be 0x80000000, but Python treats
|
|
# that as a negative number.
|
|
assert N >= 0
|
|
assert N <= 0x7fffffff
|
|
|
|
return int((self.__rng.getUint31() * N) >> 31)
|
|
|
|
def choice(self, seq):
|
|
"""returns a random element from seq"""
|
|
return seq[self.__rand(len(seq))]
|
|
|
|
def shuffle(self, x):
|
|
"""randomly shuffles x in-place"""
|
|
for i in range(len(x) - 1, 0, -1):
|
|
# pick an element in x[:i+1] with which to exchange x[i]
|
|
j = int(self.__rand(i+1))
|
|
x[i], x[j] = x[j], x[i]
|
|
|
|
def randrange(self, start, stop=None, step=1):
|
|
"""randrange([start,] stop[, step])
|
|
same as choice(range(start, stop[, step])) without construction
|
|
of a list"""
|
|
## this was lifted from Python2.2's random.py
|
|
# This code is a bit messy to make it fast for the
|
|
# common case while still doing adequate error checking
|
|
istart = int(start)
|
|
if istart != start:
|
|
raise ValueError("non-integer arg 1 for randrange()")
|
|
if stop is None:
|
|
if istart > 0:
|
|
return self.__rand(istart)
|
|
raise ValueError("empty range for randrange()")
|
|
istop = int(stop)
|
|
if istop != stop:
|
|
raise ValueError("non-integer stop for randrange()")
|
|
if step == 1:
|
|
if istart < istop:
|
|
return istart + self.__rand(istop - istart)
|
|
raise ValueError("empty range for randrange()")
|
|
istep = int(step)
|
|
if istep != step:
|
|
raise ValueError("non-integer step for randrange()")
|
|
if istep > 0:
|
|
n = (istop - istart + istep - 1) / istep
|
|
elif istep < 0:
|
|
n = (istop - istart + istep + 1) / istep
|
|
else:
|
|
raise ValueError("zero step for randrange()")
|
|
|
|
if n <= 0:
|
|
raise ValueError("empty range for randrange()")
|
|
return istart + istep*int(self.__rand(n))
|
|
|
|
def randint(self, a, b):
|
|
"""returns integer in [a, b]"""
|
|
assert a <= b
|
|
range = b-a+1
|
|
r = self.__rand(range)
|
|
return a+r
|
|
|
|
# since floats are involved, I would recommend not trusting
|
|
# this function for important decision points where remote
|
|
# synchronicity is critical
|
|
def random(self):
|
|
"""returns random float in [0.0, 1.0)"""
|
|
return float(self.__rng.getUint31()) / float(1 << 31)
|