单个概率检查可以使用线性概率轻松完成:
function checkWithProbability($probability=0.1, $length=10000)
{
$test = mt_rand(1, $length);
return $test<=$probability*$length;
}
例如,这将产生:
for($i=0; $i<10; $i++)
{
var_dump(checkWithProbability(1/3));
}
类似这样:
布尔(false)
布尔(true)
布尔(false)
布尔(false)
布尔(false)
布尔(false)
布尔(false)
布尔(false)
布尔(true)
布尔(false)
您可以使用这个原则来以所需的概率检查您的边缘:
function checkWithSet(array $set, $length=10000)
{
$left = 0;
foreach($set as $num=>$right)
{
$set[$num] = $left + $right*$length;
$left = $set[$num];
}
$test = mt_rand(1, $length);
$left = 1;
foreach($set as $num=>$right)
{
if($test>=$left && $test<=$right)
{
return $num;
}
$left = $right;
}
return null;//debug, no event realized
}
这个想法是利用几何概率 - 即将某条线段分为相应长度的几段,然后检查我们的随机数属于哪一部分。
0.75 0.9
| |
V V
*--------*--*-----*-*--*--* <-- (长度)
^ ^ ^ ^ ^
| | | | |
0 0.4 0.5 0.8 1
示例:
$set = [
1 => 0.4,
2 => 0.1,
3 => 0.25,
4 => 0.05,
5 => 0.1,
6 => 0.1
];
for($i=0; $i<10; $i++)
{
var_dump(checkWithSet($set));
}
结果如下:
int(1)
int(2)
int(2)
int(6)
int(3)
int(1)
int(1)
int(6)
int(1)
int(1)
您可以增加$length
- 理论上,这将提高随机检查的“质量”,但这并不是太容易的事情 - 因为mt_rand()
使用伪随机生成器Mersenne Twister(在理想情况下,这不是真正的线性概率)