当前位置: 首页 > 面试题库 >

如何在PHP中对数组和数据进行排序?

漆雕兴平
2023-03-14
问题内容

该问题旨在作为有关PHP中数组排序问题的参考。容易想到您的特定案例是独特的,值得提出新的问题,但是实际上大多数都是此页面上一种解决方案的细微变化。

如果您的问题作为该问题的重复而被关闭,请仅在您可以解释为什么该问题与以下所有内容明显不同时才要求重新提出您的问题。

如何在PHP中对数组排序?
如何在PHP中对 复杂 数组进行排序?
如何在PHP中对对象数组进行排序?

有关使用PHP现有功能的实用答案,请参见1 .;有关排序算法的学术详细答案(PHP函数实现的,以及对于真正非常复杂的情况 可能 需要的答案),请参见2。


问题答案:

基本的一维数组

$array = array(3, 5, 2, 8);

适用的排序功能:

  • sort
  • rsort
  • asort
  • arsort
  • natsort
  • natcasesort
  • ksort
  • krsort

两者之间的区别仅在于是否保留键-值关联(“ a“函数),是否按从低到高或反向(”r“)排序,对值或键进行排序(” k“)以及如何比较值(“nat”与正常)。

多维数组,包括对象数组

$array = array(
    array('foo' => 'bar', 'baz' => 42),
    array('foo' => ...,   'baz' => ...),
    ...
);

如果要按$array每个条目的键’foo’ 进行排序,则需要一个 自定义比较函数
。上面sort和相关的函数处理简单的值,他们知道如何比较和排序。PHP不会简单地“知道”如何处理 复杂的值, 例如array('foo' => 'bar', 'baz' => 42);所以你需要告诉它。

为此,您需要创建一个 比较函数
。该函数需要两个元素,并且0如果这些元素被认为相等,则必须返回,该值要比0第一个值低时要低,而比第一个值高时要0高。这就是所需要的:

function cmp(array $a, array $b) {
    if ($a['foo'] < $b['foo']) {
        return -1;
    } else if ($a['foo'] > $b['foo']) {
        return 1;
    } else {
        return 0;
    }
}

通常,您将需要使用匿名函数作为回调。如果要使用方法或静态方法,请参见在PHP中指定回调的其他方法。

然后,您可以使用以下功能之一:

  • usort
  • uasort
  • uksort

同样,它们只是在保留键值关联以及按值或键排序方面有所不同。阅读他们的文档以了解详细信息。

用法示例:

usort($array, 'cmp');

usort将从数组中取出两项,并cmp用它们调用函数。因此cmp()将与$aas array('foo' => 'bar', 'baz' => 42)$bas 一起调用array('foo' => ..., 'baz' => ...)。然后,函数返回到usort哪个值较大或它们是否相等。usort重复此过程,为传递不同的值$a$b直到对数组排序为止。该cmp函数将被调用多次,
至少 被调用的次数与中的值的调用次数相同,并且每次调用的值的$array组合不同。$a``$b

要习惯这个想法,请尝试以下操作:

function cmp($a, $b) {
    echo 'cmp called with $a:', PHP_EOL;
    var_dump($a);
    echo 'and $b:', PHP_EOL;
    var_dump($b);
}

您要做的就是定义一种自定义方式来比较两个项目,这就是您所需要的。这适用于各种价值。

顺便说一下,这适用于任何值,这些值不必是复杂的数组。如果您要进行自定义比较,则也可以对简单的数字数组进行比较。

sort 按引用排序,不返回任何有用的信息!

请注意,数组 在适当的位置 排序,您无需将返回值分配给任何html" target="_blank">对象。$array = sort($array)将用替换该数组true,而不用已排序的数组。只是sort($array);作品。

自定义数值比较

如果要按baz数字键进行排序,则需要做的是:

function cmp(array $a, array $b) {
    return $a['baz'] - $b['baz'];
}

多亏了 数学运算的功率, 它返回的值$a小于,等于0还是大于0,具体取决于是小于,等于还是大于$b

请注意,这不适用于float值,因为它们会降低为int并失去精度。使用显式-101返回值。

对象

如果您有一个对象数组,则其工作方式相同:

function cmp($a, $b) {
    return $a->baz - $b->baz;
}

功能

您可以在比较函数内做任何需要的事情,包括调用函数:

function cmp(array $a, array $b) {
    return someFunction($a['baz']) - someFunction($b['baz']);
}

弦乐

第一个字符串比较版本的快捷方式:

function cmp(array $a, array $b) {
    return strcmp($a['foo'], $b['foo']);
}

strcmp不正是要的是什么的cmp在这里,它返回-101

飞船操作员

PHP
7引入了spaceship运算符,该运算符统一并简化了与跨类型比较相比相等/较小/​​较大的操作:

function cmp(array $a, array $b) {
    return $a['foo'] <=> $b['foo'];
}

按多个字段排序

如果要主要按排序foo,但如果foo两个元素相等,则按baz

function cmp(array $a, array $b) {
    if (($cmp = strcmp($a['foo'], $b['foo'])) !== 0) {
        return $cmp;
    } else {
        return $a['baz'] - $b['baz'];
    }
}

对于熟悉的人,这等效于带有的SQL查询ORDER BY foo, baz。另请参见此非常简洁的速记版本,以及[如何为任意数量的键动态创建此类比较功能。

整理成手动,静态订单

如果要将元素按“手动顺序”排序,例如 “ foo”,“ bar”,“ baz”

function cmp(array $a, array $b) {
    static $order = array('foo', 'bar', 'baz');
    return array_search($a['foo'], $order) - array_search($b['foo'], $order);
}

对于上述所有情况,如果您使用的是PHP 5.3或更高版本(确实应该使用),请对较短的代码使用匿名函数,并避免使另一个全局函数随处可见:

usort($array, function (array $a, array $b) { return $a['baz'] - $b['baz']; });

这就是对复杂的多维数组进行排序的简单方式。再次,只考虑在 教授PHP时如何分辨两个项目中的哪个“更大” ;让PHP进行实际排序。

同样对于以上所有内容,要在升序和降序之间切换,只需交换$a$b参数即可。例如:

return $a['baz'] - $b['baz']; // ascending
return $b['baz'] - $a['baz']; // descending

然后是奇特的array_multisort,可让您根据另一个数组对一个数组进行排序:

$array1 = array( 4,   6,   1);
$array2 = array('a', 'b', 'c');

这里的预期结果将是:

$array2 = array('c', 'a', 'b');  // the sorted order of $array1

使用array_multisort到那里:

array_multisort($array1, $array2);

从PHP 5.5.0开始,您可以array_column从多维数组中提取一列并对该列进行排序:

array_multisort(array_column($array, 'foo'), SORT_DESC, $array);

从PHP 7.0.0开始,您还可以从对象数组中提取属性。

如果您遇到更常见的情况,请随时编辑此答案。



 类似资料:
  • 问题内容: 我已将CSV数据加载到多维数组中。这样,每个“行”都是一条记录,每个“列”都包含相同类型的数据。我正在使用下面的功能加载我的CSV文件。 我需要能够指定要排序的列,以便重新排列行。其中一列包含格式为的日期信息,我希望能够以最新日期为第一行进行排序。 问题答案: 您可以使用array_multisort() 尝试这样的事情: 对于PHP> = 5.5.0,只需提取列进行排序即可。无需循环

  • 问题内容: 我有这种格式的数组: 我想按日期的升序对它们进行排序(基于月,日和年)。最好的方法是什么? 最初,电子邮件是以MySQL日期格式获取的,因此我有可能将阵列设置为以下状态: 也许当它采用这种格式时,我可以遍历它们,删除所有(连字符)标记,使它们保留为整数,使用它们对它们进行排序,然后再次遍历它们以对其进行排序?如果有另一种方法,我会比较喜欢,因为我将为此每个用户执行3次循环。 谢谢。 编

  • 问题内容: 我有这种格式的数组: 如何按该字段的降序对该格式的数组排序?是否有内置功能? 问题答案: 使用并提供您自己的功能进行订购,例如

  • 问题内容: 我有一个由数组组成的数组。我想通过子数组的属性对父数组进行排序。这是一个例子 我想通过在子数组中升序对父数组进行排序,因此在这种情况下,结果将是子数组颠倒(,)。使用众多的PHP排序函数中的任何一种,是否可能? 问题答案: 您可以将usort函数用作:

  • 问题内容: 我的数组不包含任何字符串。但是它包含对象引用。每个对象引用都通过toString方法返回名称,id,作者和发布者。 现在,我需要按名称对对象数组进行排序。我知道如何排序,但是我不知道如何从对象中提取名称并对它们进行排序。 问题答案: 你有两种方法可以使用Arrays实用程序类 实现一个Comparator并将数组与比较器一起传递给sort方法,该方法将其作为第二个参数。 在对象所属的类

  • 我有一个JSON数组: 结果为“数据”: 我怎么能有一个升序按“datesurder”? THX