按字母排序,然后按数字排序 Laravel 集合

问题描述

我正在寻找一种对集合进行排序的方法,即以字母开头的名称位于顶部,然后以数字开头的名称。例如:

$collection = collect([
    ['name' => 'b','symbol' => '#'],['name' => '2a','symbol' => '$'],['name' => '1','symbol' => '@'],['name' => 'a','symbol' => '%']
]);

上面的集合应该是这样排序的:

[
    [
        "name" => "a","symbol" => "%",],[
        "name" => "b","symbol" => "#",[
        "name" => "1","symbol" => "@",[
        "name" => "2a","symbol" => "$",]

但这就是我使用 sortBy 方法对其进行排序时得到的结果:

$collection->sortBy('name')->values()->all();
[
    [
        "name" => "1",[
        "name" => "a",]

任何想法如何对这个集合进行排序,以便以字母开头的名称排在第一位。谢谢

解决方法

您需要定义自己的自定义比较器函数来使用 sort 对这些集合对象进行排序。

通过检查它们都是字母来比较两个名称。如果两者都是字母,那么使用 strcasecmp 进行通常的字符串比较就足够了。如果它们中的任何一个是字母表,则通过返回值 -1 将它们推到更高的等级,这意味着按排序顺序放在上面。如果两者都是数字或字母数字,请再次使用 strcasecmp

<?php

$collection = collect([
    ['name' => 'b','symbol' => '#'],['name' => '2a','symbol' => '$'],['name' => '1','symbol' => '@'],['name' => 'a','symbol' => '%']
]);

$collection = $collection->sort(function($a,$b){
    $a_is_alphabet = preg_match('/^[a-zA-Z]+$/',$a['name']) === 1;
    $b_is_alphabet = preg_match('/^[a-zA-Z]+$/',$b['name']) === 1;

    if($a_is_alphabet && $b_is_alphabet){
        return strcasecmp($a['name'],$b['name']);
    }elseif($a_is_alphabet){
        return -1;
    }elseif($b_is_alphabet){
        return 1;
    }

    return strcasecmp($a['name'],$b['name']);
});

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...