gpt4 book ai didi

algorithm - 具有指定结果概率的随机值样本

转载 作者:塔克拉玛干 更新时间:2023-11-03 03:44:31 25 4
gpt4 key购买 nike

假设我们有四个符号 - 'a''b''c''d'。我们还给出了这些符号出现在函数输出中的四个给定概率 - P1P2P3P4 (其总和等于 1)。如何实现一个函数来生成其中三个符号的随机样本,这样生成的符号以指定的概率出现在其中?

示例:'a''b''c''d' 具有9/308/307/306/30 的概率。该函数输出这四个符号中任意三个的各种随机样本:'abc''dca''bad' 等等.我们多次运行这个函数,计算每个符号在其输出中出现的次数。最后,'a' 存储的计数值除以符号输出总量应该收敛到 9/30,对于 'b'8/30'c'7/30,以及 'd'6/30

例如该函数生成 10 个输出:

adc
dab
bca
dab
dba
cab
dcb
acd
cab
abc

这 30 个符号中包含 9 个 'a'、8 个 'b'、7 个 'c' 和 6 个'。当然,这是一个理想主义的例子,因为只有当样本数量大得多时值才会收敛 - 但它应该能传达这个想法。

显然,这一切只有在两个概率都不大于 1/3 时才有可能,因为每个样本输出总是包含三个不同的符号。如果无法满足提供的值,则函数进入无限循环或以其他方式表现不稳定是可以的。

注意:该函数显然应该使用 RNG,但在其他方面应该是无状态的。除了 RNG 状态之外,每个新的调用都应该独立于之前的任何调用。

编辑:即使描述提到从 4 个值中选择 3 个值,理想情况下该算法应该能够处理任何样本量。

最佳答案

您的问题未确定。

如果我们为我们允许的三个字母组成的每个字符串分配一个概率,p(abc)、p(abd)、p(acd) 等 xtc,我们可以生成一系列等式

eqn1: p(abc) + p(abd) + ... others with a "a" ... = p1
...
...
eqn2: p(abd) + p(acd) + ... others with a "d" ... = p4

这比方程式有更多的未知数,所以有很多求解方法。找到解决方案后,通过您选择的任何方法(如果您是我,请使用单纯形算法),使用@alestanis 描述的轮盘赌方法从每个字符串的概率中抽样。

from numpy import *

# using cvxopt-1.1.5
from cvxopt import matrix, solvers

###########################
# Functions to do some parts

# function to find all valid outputs
def perms(alphabet, length):
if length == 0:
yield ""
return
for i in range(len(alphabet)):
val1 = alphabet[i]
for val2 in perms(alphabet[:i]+alphabet[i+1:], length-1):
yield val1 + val2


# roulette sampler
def roulette_sampler(values, probs):
# Create cumulative prob distro
probs_cum = [sum(probs[:i+1]) for i in range(n_strings)]
def fun():
r = random.rand()
for p,s in zip(probs_cum, values):
if r < p:
return s
# in case of rounding error
return values[-1]
return fun


############################
# Main Part



# create list of all valid strings

alphabet = "abcd"
string_length = 3
alpha_probs = [string_length*x/30. for x in range(9,5,-1)]

# show probs
for a,p in zip(alphabet, alpha_probs):
print "p("+a+") =",p




# all valid outputs for this particular case
strings = [perm for perm in perms(alphabet, string_length)]
n_strings = len(strings)

# constraints from probabilities p(abc) + p(abd) ... = p(a)
contains = array([[1. if s.find(a) >= 0 else 0. for a in alphabet] for s in strings])
#both = concatenate((contains,wons), axis=1).T # hacky, but whatever
#A = matrix(both)
#b = matrix(alpha_probs + [1.])
A = matrix(contains.T)
b = matrix(alpha_probs)

#also need to constrain to [0,1]
wons = array([[1. for s in strings]])
G = matrix(concatenate((eye(n_strings),wons,-eye(n_strings),-wons)))
h = matrix(concatenate((ones(n_strings+1),zeros(n_strings+1))))

## target matricies for approx KL divergence
# uniform prob over valid outputs
u = 1./len(strings)
P = matrix(eye(n_strings))
q = -0.5*u*matrix(ones(n_strings))
# will minimise p^2 - pq for each p val equally


# Do convex optimisation
sol = solvers.qp(P,q,G,h,A,b)
probs = array(sol['x'])

# Print ouput
for s,p in zip(strings,probs):
print "p("+s+") =",p
checkprobs = [0. for char in alphabet]
for a,i in zip(alphabet, range(len(alphabet))):
for s,p in zip(strings,probs):
if s.find(a) > -1:
checkprobs[i] += p
print "p("+a+") =",checkprobs[i]
print "total =",sum(probs)

# Create the sampling function
rndstring = roulette_sampler(strings, probs)


###################
# Verify

print "sampling..."
test_n = 1000
output = [rndstring() for i in xrange(test_n)]

# find which one it is
sampled_freqs = []
for char in alphabet:
n = 0
for val in output:
if val.find(char) > -1:
n += 1
sampled_freqs += [n]

print "plotting histogram..."
import matplotlib.pyplot as plt
plt.bar(range(0,len(alphabet)),array(sampled_freqs)/float(test_n), width=0.5)
plt.show()

编辑:Python 代码

关于algorithm - 具有指定结果概率的随机值样本,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13023787/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com