这个问题实际上是来自SO的另一个问题,我想稍微扩展一下。
在PHP中使用关联数组是可以对其值进行排序,但是在值等于保留原始键顺序的情况下,使用一个(或多个)PHP的内置排序函数?
这是我用来测试可能的解决方案的脚本(还没有找到):
<?php
header('Content-type: text/plain');
for($i=0;$i<10;$i++){
$arr['key-'.$i] = rand(1,5)*10;
}
uasort($arr, function($a, $b){
// sort condition may go here //
// Tried: return ($a == $b)?1:($a - $b); //
// Tried: return $a >= $b; //
});
print_r($arr);
?>
陷阱:因为这些键是在原始数组中排序的,所以请不要试图按键进行任何排序以恢复原始顺序。我用它们做了例子,命令更容易在输出中直观地检查它们的顺序。
答案 0 :(得分:28)
自PHP does not support stable sort after PHP 4.1.0起,您需要编写自己的函数。
这似乎符合您的要求:http://www.php.net/manual/en/function.usort.php#38827
正如手册所说,“如果两个成员相等,那么它们在排序数组中的顺序是不确定的。”这意味着使用的排序不是“稳定的”,可能会改变比较相等的元素的顺序。
有时你确实需要稳定的排序。例如,如果您按一个字段对列表进行排序,然后再按另一个字段对其进行排序,但不希望丢失前一个字段的排序。在这种情况下,最好使用带有比较函数的usort来考虑两个字段,但如果你不能这样做,那么使用下面的函数。它是一种合并排序,保证了O(n * log(n))复杂度,这意味着即使使用较大的列表它也会保持相当快(与bubblesort和插入排序不同,它们是O(n ^ 2))。
<?php
function mergesort(&$array, $cmp_function = 'strcmp') {
// Arrays of size < 2 require no action.
if (count($array) < 2) return;
// Split the array in half
$halfway = count($array) / 2;
$array1 = array_slice($array, 0, $halfway);
$array2 = array_slice($array, $halfway);
// Recurse to sort the two halves
mergesort($array1, $cmp_function);
mergesort($array2, $cmp_function);
// If all of $array1 is <= all of $array2, just append them.
if (call_user_func($cmp_function, end($array1), $array2[0]) < 1) {
$array = array_merge($array1, $array2);
return;
}
// Merge the two sorted arrays into a single sorted array
$array = array();
$ptr1 = $ptr2 = 0;
while ($ptr1 < count($array1) && $ptr2 < count($array2)) {
if (call_user_func($cmp_function, $array1[$ptr1], $array2[$ptr2]) < 1) {
$array[] = $array1[$ptr1++];
}
else {
$array[] = $array2[$ptr2++];
}
}
// Merge the remainder
while ($ptr1 < count($array1)) $array[] = $array1[$ptr1++];
while ($ptr2 < count($array2)) $array[] = $array2[$ptr2++];
return;
}
?>
另外,您可能会发现this forum thread有趣。
答案 1 :(得分:11)
array_multisort
派上用场,只使用有序范围作为第二个数组($order
只是临时的,它用于按原始顺序排序第一个数组的等效项):
$a = [
"key-0" => 5,
"key-99" => 3,
"key-2" => 3,
"key-3" => 7
];
$order = range(1,count($a));
array_multisort($a, SORT_ASC, $order, SORT_ASC);
var_dump($a);
<强>输出强>
array(4) {
["key-99"]=>
int(3)
["key-2"]=>
int(3)
["key-0"]=>
int(5)
["key-3"]=>
int(7)
}
我使用带有非有序键的测试数据来证明它可以正常工作。尽管如此,这是测试脚本的输出:
Array
(
[key-1] => 10
[key-4] => 10
[key-5] => 20
[key-8] => 20
[key-6] => 30
[key-9] => 30
[key-2] => 40
[key-0] => 50
[key-3] => 50
[key-7] => 50
)
它只适用于预定义的比较,您不能使用自己的比较功能。可能的值(array_multisort()
的第二个参数)是:
排序类型标记:
SORT_ASC
- 按升序排序。SORT_DESC
- 对项目进行降序排序。SORT_REGULAR
- 正常比较项目(不要更改类型)SORT_NUMERIC
- 以数字方式比较项目SORT_STRING
- 将项目比较为字符串SORT_LOCALE_STRING
- 根据当前区域设置将项目比较为字符串。它使用可以使用的区域设置setlocale()
- )将项目比较为字符串
SORT_NATURAL
- 使用“自然排序”(例如natsort()
SORT_FLAG_CASE
- 可以与SORT_STRING
或SORT_NATURAL
合并(按位或)以对字符串不区分大小写进行排序
答案 2 :(得分:7)
为了将来的参考,我在Github上添加了一组内置PHP函数的稳定排序变体:https://github.com/vanderlee/PHP-stable-sort-functions,基于@Jack's的解决方案和其他一些技巧。
答案 3 :(得分:5)
为了完整起见,您还应该查看Schwartzian transform:
// decorate step
$key = 0;
foreach ($arr as &$item) {
$item = array($item, $key++); // add array index as secondary sort key
}
// sort step
asort($arr); // sort it
// undecorate step
foreach ($arr as &$item) {
$item = $item[0]; // remove decoration from previous step
}
PHP的默认排序算法适用于数组,因为:
array(1, 0) < array(2, 0); // true
array(1, 1) < array(1, 2); // true
如果您想使用自己的排序标准,也可以使用uasort()
:
// each parameter is an array with two elements
// [0] - the original item
// [1] - the array key
function mysort($a, $b)
{
if ($a[0] != $b[0]) {
return $a[0] < $b[0] ? -1 : 1;
} else {
// $a[0] == $b[0], sort on key
return $a[1] < $b[1] ? -1 : 1; // ASC
}
}
答案 4 :(得分:1)
这是一个使用它可以在usort函数中实现稳定排序的解决方案
public function sortBy(array &$array, $value_compare_func)
{
$index = 0;
foreach ($array as &$item) {
$item = array($index++, $item);
}
$result = usort($array, function($a, $b) use ($value_compare_func) {
$result = call_user_func($value_compare_func, $a[1], $b[1]);
return $result == 0 ? $a[0] - $b[0] : $result;
});
foreach ($array as &$item) {
$item = $item[1];
}
return $result;
}
答案 5 :(得分:0)
只是用一些非常具体的案例来完成答案。如果$array
的数组键是默认值,那么简单array_values(asort($array))
就足够了(例如按升序排列)
答案 6 :(得分:0)
作为稳定排序的一种解决方法:
<?php
header('Content-type: text/plain');
for ($i = 0;$i < 10;$i++)
{
$arr['key-' . $i] = rand(1, 5) * 10;
}
uksort($arr, function ($a, $b) use ($arr)
{
if ($arr[$a] === $arr[$b]) return array_search($a, array_keys($arr)) - array_search($b, array_keys($arr));
return $arr[$a] - $arr[$b];
});
print_r($arr);