我正在尝试从0-n生成一个随机数组,然后进行随机播放(但要确保键和值不匹配)。
例如:
0 => 3
1 => 2
2 => 4
3 => 0
4 => 1
请注意,键和值均为0-4,但键和值均不相同。
有什么想法吗?
答案 0 :(得分:13)
更短的解决方案:
$random_number_array = range(0, 100);
shuffle($random_number_array );
$random_number_array = array_slice($random_number_array ,0,10);
print_r($random_number_array);
结果将是:
[0] => 53
[1] => 6
[2] => 16
[3] => 59
[4] => 8
[5] => 18
[6] => 62
[7] => 39
[8] => 22
[9] => 26
答案 1 :(得分:4)
$max = 5;
$done = false;
while(!$done){
$numbers = range(0, $max);
shuffle($numbers);
$done = true;
foreach($numbers as $key => $val){
if($key == $val){
$done = false;
break;
}
}
}
答案 2 :(得分:2)
天真的解决方案:
$n = 10;
$rands = array();
for($i=0; $i<$n;$i++) {
$ok = false;
while(!$ok) {
$x=mt_rand(0,$n-1);
$ok = !in_array($x, $rands) && $x != $i;
}
$rands[$i]=$x;
}
var_dump($rands);
高效的解决方案:
$n = 100;
$numbers = range(0, $n-1);
$rands = array();
for ($i=0; $i < $n; $i++) {
$ok = false;
while (!$ok) {
$x = array_rand($numbers);
$ok = !in_array($numbers[$x], $rands) && $numbers[$x] != $i;
}
$rands[$i] = $numbers[$x];
unset($numbers[$x]);
}
var_dump($rands);
编辑: s / rand / mt_rand /
编辑#2:如@AMayer所述,两种解决方案都可能陷入僵局。我有所纠正。
答案 3 :(得分:2)
我相信,这是一个相当长的但非常有效的解决方案。与此处发布的其他解决方案相反,这不会死锁(除非$size<2
),并且每次一个值不适合时,这不会进行完全洗牌。相反,它只会用另一个随机值替换该值。
function unique_list($size=5) {
function all_unique($numbers) {
foreach ($numbers as $key=>$value)
if ($key==$value) return false;
return true;
}
function flip($a, $b, &$numbers) {
$numbers[$a] = $numbers[$a] + $numbers[$b];
$numbers[$b] = $numbers[$a] - $numbers[$b];
$numbers[$a] = $numbers[$a] - $numbers[$b];
}
$flip_count = 0;
$numbers = range(0,$size-1);
shuffle($numbers);
while (!all_unique($numbers)) {
foreach ($numbers as $key=>$value) {
if ($key==$value) {
flip($key, rand(0,$size-1), $numbers);
$flip_count++;
break;
}
}
}
printf("Flipped %d values\n", $flip_count);
return $numbers;
}
$list = unique_list(10);
print_r($list);
以上将打印类似于
的内容Flipped 1 value(s)
Array
(
[0] => 2
[1] => 5
[2] => 7
[3] => 9
[4] => 6
[5] => 3
[6] => 1
[7] => 8
[8] => 0
[9] => 4
)
答案 4 :(得分:0)
将range(...)
(用于首先生成原始数组),shuffle(...)
(用于对数组进行随机化)和array_intersect_assoc(...)
(用于检查结果数组)组合在一起是什么?
$numbers = range(0, 4);
do {
shuffle($numbers); // print_r($numbers);
$matchingKeyValuePairs = array_intersect_assoc(array_keys($numbers), array_values($numbers)); // print_r($matchingKeyValuePairs);
} while (! empty($matchingKeyValuePairs));
此解决方案可能会为大量元素带来一些性能问题。但是可以通过处理$matchingKeyValuePairs
的逻辑对其进行扩展。因此,虽然现在的逻辑就像“如果有matchingKeyValuePairs
,然后再试一次”,但更有效的逻辑思想是“如果有matchingKeyValuePairs
,然后从阵列中切除matchingKeyValuePairs
,将该子数组随机化(如果需要,可以多次),然后将其合并回去。”
答案 5 :(得分:0)
这将生成一个数组,其中包含10个随机数,范围从0到100:
array_map(function () {
return rand(0, 100);
}, array_fill(0, 10, null)
);
结果:
array(10) {
[0]=>
int(15)
[1]=>
int(97)
[2]=>
int(20)
[3]=>
int(64)
[4]=>
int(57)
[5]=>
int(38)
[6]=>
int(16)
[7]=>
int(53)
[8]=>
int(56)
[9]=>
int(22)
}
说明:
array_fill(0, 10, null)
将生成一个包含10个空项目的数组array_map
将回调(第一个参数)应用于它接收到的数组的每个项目(第二个参数)。在此示例中,我们只为每个数组项返回一个随机数。答案 6 :(得分:0)
这里的方法是创建一个从0到n的数字范围,然后随机播放。 然后,我们寻找键值匹配,并交换它们对。如果没有一对可以交换,则与最后一个交换,除非该项目是最后一个,然后与第一交换。
private static class CounterHolder{
int value;
}
Map<String, CounterHolder> count = new HashMap<>();
List<String> words = ...
for (String word : words) {
CounterHolder holder;
if (count.containsKey(word)) {
holder = new CounterHolder();
} else {
holder = new CounterHolder();
count.put(word, holder);
}
++holder.value;
}
在n为5时随机播放的示例:
<?php
$n = 5;
$values = range(0, $n);
shuffle($values);
$matched = array_filter($values, function($v, $k) {return $k === $v;}, ARRAY_FILTER_USE_BOTH);
foreach(array_chunk($matched, 2) as list($a, $b)) {
if($b === null) {
$swap_key = array_key_last($values);
if($swap_key == $a) {
$swap_key = array_key_first($values);
}
list($values[$a], $values[$swap_key]) = [$values[$swap_key], $a];
} else {
list($values[$a], $values[$b]) = [$b, $a];
}
}
在这里,我们有键的匹配项:
array (
0 => 0,
1 => 5,
2 => 1,
3 => 3,
4 => 4,
5 => 2,
)
因此,我们将键0和3以及键4的值与最后一个交换。
0, 3 and 4.
({array (
0 => 3,
1 => 5,
2 => 1,
3 => 0,
4 => 2,
5 => 4,
)
可以换成0,因为这里给出了范围。我将其保留为更明确。)