
本教程深入探讨了如何在php中对复杂的多维数组进行多级自定义排序。我们将详细演示如何使用`usort`函数,根据主键(如`counted`)进行降序排列,并在主键值相同时,再根据次级键(如`placement`)进行升序排列,以满足特定的数据组织需求。
在PHP开发中,我们经常需要处理包含复杂结构的数据集,例如由多个关联数组组成的多维数组。当这些数据需要按照特定的业务逻辑进行组织和展示时,单一键的排序往往无法满足需求。此时,多级自定义排序就显得尤为重要,它允许我们定义多个排序条件及其优先级。
场景描述
假设我们有一个包含用户数据(或任何记录)的数组,每个元素都是一个关联数组,其中包含id、placement、counted和user等字段。我们的目标是:
- 首先,根据counted字段的值进行降序排序(即counted值大的排在前面)。
- 其次,如果两个或多个元素的counted值相同,则根据placement字段的值进行升序排序(即placement值小的排在前面)。
以下是原始数据示例:
$array = [
[
'id' => 1,
'placement' => 8,
'counted' => 3,
'user' => ['name' => 'foo'],
],
[
'id' => 2,
'placement' => 5,
'counted' => 3,
'user' => ['name' => 'bar'],
],
[
'id' => 3,
'placement' => 1,
'counted' => 2,
'user' => ['name' => 'foobar'],
]
];根据上述排序规则,我们期望得到的排序结果是:
立即学习“PHP免费学习笔记(深入)”;
$array = [
[
'id' => 2,
'placement' => 5,
'counted' => 3,
'user' => ['name' => 'bar'],
],
[
'id' => 1,
'placement' => 8,
'counted' => 3,
'user' => ['name' => 'foo'],
],
[
'id' => 3,
'placement' => 1,
'counted' => 2,
'user' => ['name' => 'foobar'],
]
];可以看到,counted字段按3, 3, 2降序排列。对于counted同为3的记录,placement字段按5, 8升序排列。
使用 usort 实现自定义多级排序
PHP提供了 usort() 函数,允许我们使用自定义的比较函数来对数组进行排序。usort() 函数接受两个参数:要排序的数组和比较函数。比较函数接收两个元素作为参数,并根据它们的相对顺序返回:
本系统经过多次升级改造,系统内核经过多次优化组合,已经具备相对比较方便快捷的个性化定制的特性,用户部署完毕以后,按照自己的运营要求,可实现快速定制会费管理,支持在线缴费和退费功能财富中心,管理会员的诚信度数据单客户多用户登录管理全部信息支持审批和排名不同的会员级别有不同的信息发布权限企业站单独生成,企业自主决定更新企业站信息留言、询价、报价统一管理,分系统查看分类信息参数化管理,支持多样分类信息,
- 1:如果第一个元素应排在第二个元素之后。
- -1:如果第一个元素应排在第二个元素之前。
- 0:如果两个元素被认为是相等的(相对顺序不变)。
为了实现多级排序,我们需要在比较函数中构建一个逻辑链,首先比较主排序键,如果主排序键相同,则继续比较次排序键,依此类推。
以下是实现上述排序逻辑的 usort 示例代码:
1,
'placement' => 8,
'counted' => 3,
'user' => ['name' => 'foo'],
],
[
'id' => 2,
'placement' => 5,
'counted' => 3,
'user' => ['name' => 'bar'],
],
[
'id' => 3,
'placement' => 1,
'counted' => 2,
'user' => ['name' => 'foobar'],
]
];
usort($array, function ($a, $b) {
// 步骤1:根据 'counted' 字段进行降序排序
// 如果 $a 的 'counted' 值小于 $b,则 $a 应该排在 $b 之后 (返回 1)
if ($a['counted'] < $b['counted']) {
return 1;
}
// 如果 $a 的 'counted' 值大于 $b,则 $a 应该排在 $b 之前 (返回 -1)
if ($a['counted'] > $b['counted']) {
return -1;
}
// 步骤2:如果 'counted' 字段值相同,则根据 'placement' 字段进行升序排序
// 如果 $a 的 'placement' 值大于 $b,则 $a 应该排在 $b 之后 (返回 1)
if ($a['placement'] > $b['placement']) {
return 1;
}
// 如果 $a 的 'placement' 值小于 $b,则 $a 应该排在 $b 之前 (返回 -1)
if ($a['placement'] < $b['placement']) {
return -1;
}
// 步骤3:如果所有比较字段都相同,则认为两者相等 (返回 0)
return 0;
});
echo '';
print_r($array);
echo '
';
?>代码解析:
-
主排序条件 (counted 降序):
- if ($a['counted']
- if ($a['counted'] > $b['counted']) { return -1; }: 如果 $a 的 counted 值比 $b 大,说明 $a 应该排在 $b 前面,所以返回 -1。
- 如果以上两个条件都不满足,说明 $a['counted'] 和 $b['counted'] 相等,此时进入次级排序。
-
次排序条件 (placement 升序):
- if ($a['placement'] > $b['placement']) { return 1; }: 在 counted 值相同的前提下,如果 $a 的 placement 值比 $b 大,说明 $a 应该排在 $b 后面(因为是升序),所以返回 1。
- if ($a['placement']
- 如果以上所有条件










