我有这个:
<?php $n = rand(1,1600); echo $n ?>
我想从随机数中排除,比如 234、1578、763、1274 和其他数字。我该怎么做?
<?php
while( in_array( ($n = mt_rand(1,1600)), array(234, 1578 ,763 , 1274) ) );
像这样试试
do {
$n = rand(1,1600);
} while(in_array($n, array(234, 1578 ,763 , 1274 ));
echo $n;
检查该数字是否是您不想要的数字,如果它是获取一个新的随机数。
function getRandomNumber() {
do {
$n = mt_rand(1,1600);
} while(in_array($n, array(234,1578, 763, 1274)));
return $n;
}
始终使用强大的加密算法来生成随机数:
/**
* @param int $from From number
* @param int $to To number
* @param array $excluded Additionally exclude numbers
* @return int
*/
function randomNumber($from, $to, array $excluded = [])
{
$func = function_exists('random_int') ? 'random_int' : 'mt_rand';
do {
$number = $func($from, $to);
} while (in_array($number, $excluded, true));
return $number;
}
var_dump(randomNumber(1, 100));
var_dump(randomNumber(1, 10, [5, 6, 7, 8]));
var_dump(randomNumber(1, 100, range(10, 90)));
如果使用多个 PHP 版本,我还建议使用paragonie/random_compat库以实现兼容性。
或者避免使用随机(可能无限)运行时间进行循环:
/**
* Returns a random integer between $min and $max (inclusive) and
* excludes integers in $exarr, returns false if no such number
* exists.
*
* $exarr is assumed to be sorted in increasing order and each
* element should be unique.
*/
function random_exclude($min, $max, $exarr = array()) {
if ($max - count($exarr) < $min) {
return false;
}
// $pos is the position that the random number will take
// of all allowed positions
$pos = rand(0, $max - $min - count($exarr));
// $num being the random number
$num = $min;
// while $pos > 0, step to the next position
// and decrease if the next position is available
for ($i = 0; $i < count($exarr); $i += 1) {
// if $num is on an excluded position, skip it
if ($num == $exarr[$i]) {
$num += 1;
continue;
}
$dif = $exarr[$i] - $num;
// if the position is after the next excluded number,
// go to the next excluded number
if ($pos >= $dif) {
$num += $dif;
// -1 because we're now at an excluded position
$pos -= $dif - 1;
} else {
// otherwise, return the free position
return $num + $pos;
}
}
// return the number plus the open positions we still had to go
return $num + $pos;
}
此函数选择一个随机位置并遍历排除数组以找到空闲位置。它的运行时间取决于要排除的数字量。如果您想排除某些范围,您可能需要调整算法以考虑到这一点。
另一个解决方案可能如下:
function random_number($min, $max, $exclude)
{
$number = rand($min, $max);
if(in_array($number, $exclude))
{
random_number($min, $max, $exclude);
} else {
return $number;
}
}
$number = random_number(1,10, [2,5,6]);
随着“黑名单”整数的数量接近整数范围的数量,接受@regenschein 的建议变得越来越有说服力。
非迭代方法可能如下所示:
$range = range(1, 1600);
$blacklist = [234, 1578, 763, 1274]; // 4 blacklisted versus 1600 full range is NOT compelling
$valids = array_diff($range, $blacklist);
echo array_values($valids)[rand(0, count($valids) - 1)];
// or
echo $valids[array_rand($valids)];
// the two approaches use different randomizers
或者,如果您也很乐意改组,您可以这样做:
$blacklist = [234, 1578, 763, 1274];
$range = range(1, 1600);
$valids = array_diff($range, $blacklist);
shuffle($valids);
echo $valids[0];
*array_diff()
如果您想传递多个黑名单数组,请注意特别好 - 只需用逗号分隔它们。
例如:
var_export($valids = array_diff(range(1, 100), range(5, 50), range(61, 99), [55]));
输出:
array (
0 => 1,
1 => 2,
2 => 3,
3 => 4,
50 => 51,
51 => 52,
52 => 53,
53 => 54,
55 => 56,
56 => 57,
57 => 58,
58 => 59,
59 => 60,
99 => 100,
)
我知道这有点老了,但我认为 op 试图做的是对整数进行洗牌。如果是这样,以下方法更好
$array = array(1,2,3,4,5,6,7);
shuffle($array);
此代码将随机化数组的确切元素的顺序而不重复,并在数组本身内返回结果。
您可以创建一个包含有效数字的数组。
然后,您的随机数生成应该将索引返回到该数组中。
如果您没有太多要排除的号码,则在找到不需要的号码时重试会更容易、更快捷:
$n = 0;
while (in_array($n, array(0, 234, 1578 ,763 , 1274))) {
$n = rand(1,1600);
}
echo $n;