如何在PHP中实现多维数组的多级自定义排序

本教程深入探讨了如何在php中对复杂的多维数组进行多级自定义排序。我们将详细演示如何使用`usort`函数,根据主键(如`counted`)进行降序排列,并在主键值相同时,再根据次级键(如`placement`)进行升序排列,以满足特定的数据组织需求。

在PHP开发中,我们经常需要处理包含复杂结构的数据集,例如由多个关联数组组成的多维数组。当这些数据需要按照特定的业务逻辑进行组织和展示时,单一键的排序往往无法满足需求。此时,多级自定义排序就显得尤为重要,它允许我们定义多个排序条件及其优先级。

场景描述

假设我们有一个包含用户数据(或任何记录)的数组,每个元素都是一个关联数组,其中包含id、placement、counted和user等字段。我们的目标是:

  1. 首先,根据counted字段的值进行降序排序(即counted值大的排在前面)。
  2. 其次,如果两个或多个元素的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'],
    ]
];

根据上述排序规则,我们期望得到的排序结果是:

$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 '
'; ?>

代码解析:

  1. 主排序条件 (counted 降序):

    • if ($a['counted']
    • if ($a['counted'] > $b['counted']) { return -1; }: 如果 $a 的 counted 值比 $b 大,说明 $a 应该排在 $b 前面,所以返回 -1。
    • 如果以上两个条件都不满足,说明 $a['counted'] 和 $b['counted'] 相等,此时进入次级排序。
  2. 次排序条件 (placement 升序):

    • if ($a['placement'] > $b['placement']) { return 1; }: 在 counted 值相同的前提下,如果 $a 的 placement 值比 $b 大,说明 $a 应该排在 $b 后面(因为是升序),所以返回 1。
    • if ($a['placement']
    • 如果以上所有条件