在 Perl 中按多个键对哈希数组进行排序

问题描述 投票:0回答:3

我有一个包含哈希值的数组引用(即@AOH):

$arr_ref = [ { 'brand' => 'A',
               'supplier' => 'X',
               'PO' => '2'
              },
              { 'brand' => 'B',
                'supplier' => 'Y',
                'PO' => '1'
              },
              { 'brand' => 'B',
                'supplier' => 'X',
                'PO' => '2'
              },
              { 'brand' => 'A',
                'supplier' => 'X',
                'PO' => '1'
              },
              { 'brand' => 'B',
                'supplier' => 'X',
                'PO' => '1'
              }
];

我想根据所有三个键(即品牌、供应商和采购订单)对其进行排序。 排序顺序应该是品牌第一,然后是供应商,最后是PO。

排序后的数组引用应该是:

$arr_ref = [ { 'brand' => 'A',
                'supplier' => 'X',
                'PO' => '1'
              },
              { 'brand' => 'A',
               'supplier' => 'X',
               'PO' => '2'
              },
              { 'brand' => 'B',
                'supplier' => 'X',
                'PO' => '1'
              },
              { 'brand' => 'B',
                'supplier' => 'X',
                'PO' => '2'
              },
              { 'brand' => 'B',
                'supplier' => 'Y',
                'PO' => '1'
              },
];
perl
3个回答
57
投票

由于

<=>
cmp
返回 0 来表示相等,这是错误的,并且因为 Perl 的逻辑布尔运算符返回决定值而不是仅 0 或 1,所以按多个键排序就像将多个比较串在一起一样简单与
or
||
:

@$arr_ref = sort { $a->{brand}    cmp $b->{brand}    or 
                   $a->{supplier} cmp $b->{supplier} or 
                   $a->{PO}       <=> $b->{PO} 
                 } @$arr_ref;

我假设 PO 是一个数字字段,因此您使用

<=>
而不是
cmp


6
投票

以下应该对数组引用进行排序并将数组放回

$arr_ref

$arr_ref = [sort by_brand_supplier_PO @$arr_ref];

sub by_brand_supplier_PO {
    $a->{brand} cmp $b->{brand} ||
    $a->{supplier} cmp $b->{supplier} ||
    $a->{PO} <=> $b->{PO}
}

2
投票

您可以使用 Sort::Key::Multi,与 Sort::Key 一起分发。

在本例中,我们使用

ssikeysort
,它需要一个返回字符串、字符串和整数的块,并按该元组对值进行排序。 (
s
中的
ssi
代表字符串,
i
代表整数。)

use Sort::Key::Multi qw(ssikeysort);

@$arr_ref = ssikeysort { $_->{brand}, $_->{supplier}, $_->{PO} } @$arr_ref;

您还可以使用就地变体,它使用更少的内存:

use Sort::Key::Multi qw(ssikeysort_inplace);

ssikeysort_inplace { $_->{brand}, $_->{supplier}, $_->{PO} } @$arr_ref;
© www.soinside.com 2019 - 2024. All rights reserved.