-
Notifications
You must be signed in to change notification settings - Fork 22
/
Copy pathmyrandom.py
46 lines (42 loc) · 1.28 KB
/
myrandom.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
"""
Random sampling routines.
Here is an example:
from common.myrandom import build, weighted_sample
keys = "ABC"
weights = [1., 3., 2.]
indexed_weights = build(weights)
print keys[weighted_sample(indexed_weights)]
print keys[weighted_sample(indexed_weights)]
"""
def build(weights):
"""
Create an index of weights. Must be done prior to calling weighted_sample.
"""
indexed_weights = []
sum = 0.
for w in weights:
indexed_weights.append(sum)
sum += w
return ("indexed_weights", indexed_weights, sum)
def weighted_sample(indexed_weights):
"""
Sample an index, according to the weights in indexed_weights.
indexed_weights must be obtained from the build() functon.
Return the index, and its probability.
"""
assert len(indexed_weights) == 3
assert indexed_weights[0] == "indexed_weights"
tot = indexed_weights[2]
indexed_weights = indexed_weights[1]
from bisect import bisect
import random
v = random.random()
v *= tot
idx = bisect(indexed_weights, v)
idx -= 1
assert idx >= 0 and idx < len(indexed_weights)
if idx == len(indexed_weights) - 1:
pr = 1. * (tot - indexed_weights[idx]) / tot
else:
pr = 1. * (indexed_weights[idx+1] - indexed_weights[idx]) / tot
return idx, pr