PHP数组同值稳定排序

在 PHP 中对数组排序有很多函数,如 sort, asort, arsort, ksort, krsort, uasort 等等,但是有一个问题,如一个数组:

$test = array(1 => 0, 99 => 0, 87 => 0, 45 => 0, 67 => 0, 11 => 1, 2 => 0);

$test = array(1 => 0, 99 => 0, 87 => 0, 45 => 0, 67 => 0, 11 => 1, 2 => 0);

用 asort 函数对其进行排序,asort($test) 结果如下(转换为 json 格式):

{"67":0,"2":0,"1":0,"45":0,"87":0,"99":0,"11":1}

{"67":0,"2":0,"1":0,"45":0,"87":0,"99":0,"11":1}

可以看到排序虽然正确,值为 1 的元素排到后面去了,但是值为 0 的元素顺序跟以前不一样了,如何保证 PHP 数组同值元素排序后顺序保持不变呢?

感谢网友的回答:

PHP 的 asort 和 sort,底层是用快排实现的。无法保证同等大小的元素的顺序。要保证值为0的顺序不变?只能自己实现一个数组排序。对于数组中相等的元素,它们在排序后的顺序是未定义的。 (也即相等元素之间的顺序是不稳定的)

解决方法

1.在 PHP 官方手册上其实有答案了,直接贴出来:

If you want to keep the order when two members compare as equal, use this.

function stable_uasort(&$array, $cmp_function) {

    if(count($array) < 2) {

        return;

    }

    $halfway = count($array) / 2;

    $array1 = array_slice($array, 0, $halfway, TRUE);

    $array2 = array_slice($array, $halfway, NULL, TRUE);

    stable_uasort($array1, $cmp_function);

    stable_uasort($array2, $cmp_function);

    if(call_user_func($cmp_function, end($array1), reset($array2)) < 1) {

        $array = $array1 + $array2;

        return;

    }

    $array = array();

    reset($array1);

    reset($array2);

    while(current($array1) && current($array2)) {

        if(call_user_func($cmp_function, current($array1), current($array2)) < 1) {

            $array[key($array1)] = current($array1);

            next($array1);

        } else {

            $array[key($array2)] = current($array2);

            next($array2);

        }

    }

    while(current($array1)) {

        $array[key($array1)] = current($array1);

        next($array1);

    }

    while(current($array2)) {

        $array[key($array2)] = current($array2);

        next($array2);

    }

    return;

}

function cmp($a, $b) {

    if($a['n'] == $b['n']) {

        return 0;

    }

    return ($a['n'] > $b['n']) ? -1 : 1;

}

$a = $b = array(

    'a' => array("l" => "A", "n" => 1),

    'b' => array("l" => "B", "n" => 2),

    'c' => array("l" => "C", "n" => 1),

    'd' => array("l" => "D", "n" => 2),

    'e' => array("l" => "E", "n" => 2),

);

uasort($a, 'cmp');

print_r($a);

stable_uasort($b, 'cmp');

print_r($b);
  1. 如果使用 Composer,也可以使用 PHP 扩展包,这里有一个扩展包 vanderlee/php-stable-sort-functions 就是专门解决这个问题的。

bool StableSort::arsort ( array &$array [, int $sort_flags = SORT_REGULAR ] )

bool StableSort::asort ( array &$array [, int $sort_flags = SORT_REGULAR ] )

bool StableSort::natcasesort ( array &Sarray )

bool StableSort::natsort ( array &Sarray )

bool StableSort::uasort ( array &$array , callable $value_compare_func )

bool StableSort::uksort ( array &$array , callable $value_compare_func )

bool StableSort::usort ( array &$array , callable $value_compare_func )

推荐链接

扩展包地址:https://packagist.org/packages/vanderlee/php-stable-sort-functions

https://segmentfault.com/q/1010000007492504?_ea=1362649

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,711评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,932评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,770评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,799评论 1 271
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,697评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,069评论 1 276
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,535评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,200评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,353评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,290评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,331评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,020评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,610评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,694评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,927评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,330评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,904评论 2 341

推荐阅读更多精彩内容

  • 第三章 数组 数组是一个可以存储一组或一系列数值的变量 一个包含其他数组的数组称为多维数组。 关联数组允许使用更...
    梁烨端木阅读 655评论 0 0
  • 背景 一年多以前我在知乎上答了有关LeetCode的问题, 分享了一些自己做题目的经验。 张土汪:刷leetcod...
    土汪阅读 12,716评论 0 33
  • php usleep() 函数延迟代码执行若干微秒。 unpack() 函数从二进制字符串对数据进行解包。 uni...
    思梦PHP阅读 1,979评论 1 24
  • PHP常用函数大全 usleep() 函数延迟代码执行若干微秒。 unpack() 函数从二进制字符串对数据进行解...
    上街买菜丶迷倒老太阅读 1,346评论 0 20
  • 什么是内心的强大: 不再对失去的事物惋惜对未来惶恐, 充实眼前的生活并坦然面对一切。 不再记恨别人的负面评价, 也...
    Ferrari_ma阅读 163评论 0 0