繁体   English   中英

按值而不是模板工具包中的键对哈希引用进行排序

[英]Sort a hash reference by value instead of key in Template Toolkit

我有一个 perl 多维哈希引用,我以下列方式在 Template Toolkit 中迭代它。 有没有办法按其中一个值(例如 AVG 而不是键)对其进行排序?

   <tr>
   <% FOREACH name in payload_batting.keys.sort %>
   <td width="140"><% name.substr(0, 18);%></td>
   <td width="55"><% payload_batting.$name.atts %></td>
   <td width="55"><% payload_batting.$name.runs %></td>
   <td width="55"><% payload_batting.$name.hits %></td>
   <td width="55"><% payload_batting.$name.AVG %></td>        
   <tr>
   <% END %>

正如池上所说,你必须自己整理。 如果您想在表示层中进行排序,在 Template Toolkit 中,您可以在模板本身中作为 Perl 代码进行排序。 为此,您必须将 EVAL_PERL 设置为真值。 然后代码看起来像这样

[% PERL %]
my $h = $stash->get('payload_batting');
my @k = sort { $h->{$a}->{AVG} <=> $h->{$b}->{AVG} } keys %$h;
$stash->set('k', \@k);
[% END %]

[% FOREACH name IN k %]
<tr>
<td width="140">[% name.substr(0, 18);%]</td>
<td width="55">[% payload_batting.$name.atts %]</td>
<td width="55">[% payload_batting.$name.runs %]</td>
<td width="55">[% payload_batting.$name.hits %]</td>
<td width="55">[% payload_batting.$name.AVG %]</td>
<tr>
[% END %]

更简洁的是注册一个新的自定义虚拟方法,它可以为您进行排序(文档在这里):

首先,您必须定义为您实现键排序的虚拟方法。 这只是一个简单的实现,允许您按结构中的多个级别对嵌套散列进行排序:

use Template::Stash;

$Template::Stash::HASH_OPS->{'deep_sort'} = sub {
  my ($h, $name, $op) = @_;
  $op //= 'cmp';
  my @fields = split(/\./,$name);
  my $resolve = sub {
    my $v = $_[0];
    foreach (@fields) {
      $v = $v->{$_};
    }
    return $v;
  };
  return [sort { $resolve->($h->{$a}) cmp $resolve->($h->{$b}) } keys %$h ]
    if $op eq 'cmp';
  return [sort { $resolve->($h->{$a}) <=> $resolve->($h->{$b}) } keys %$h ]
    if $op eq '<=>';
};

然后你必须像这样在模板中引用它:

[% FOREACH name IN payload_batting.deep_sort('AVG','<=>') %]
<tr>
   <td width="140">[% name.substr(0, 18);%]</td>
   <td width="55">[% payload_batting.$name.atts %]</td>
   <td width="55">[% payload_batting.$name.runs %]</td>
   <td width="55">[% payload_batting.$name.hits %]</td>
   <td width="55">[% payload_batting.$name.AVG %]</td>
<tr>
[% END %]

具有更深数据结构的不同情况:

[%
data = {
  payload_batting => { STAT => { AVG => 4, SUM => 16 } }
}
%]

[% data.deep_sort('STAT.AVG','<=>') %]

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM