从哈希值生成组合

5

如果我在Perl中有一个哈希表

my %ranges = (                                                                                      
'--tic' => [ 0, 1, 2 ],                                                                             
'--threads' => [ 8, 16 ],                                                                           
'--level' => [ 10, 20 ]                                                                               
);                                                                     

如何生成所有组合的数组,例如

--level 10 --threads 8 --tic 0                                                                        
--level 10 --threads 8 --tic 1                                                                        
--level 10 --threads 8 --tic 2                                                                        
--level 10 --threads 16 --tic 0                                                                       
--level 10 --threads 16 --tic 1                                                                       
--level 10 --threads 16 --tic 2                                                                       
--level 20 --threads 8 --tic 0                                                                        
--level 20 --threads 8 --tic 1                                                                        
--level 20 --threads 8 --tic 2                                                                        
--level 20 --threads 16 --tic 0                                                                       
--level 20 --threads 16 --tic 1                                                                       
--level 20 --threads 16 --tic 2       

哈希表中可以有任意数量的哈希项,每个哈希项的值数组中可以有任意数量的元素。输出数组的顺序无关紧要,只需要为每个组合有1个元素,在这种情况下为3*2*2=12,但可以是任何数字。

我认为使用splice、map和foreach的一些组合应该可以解决问题,但我正在摸索中。


1
你正在尝试生成所谓的笛卡尔积(或叉积);CPAN上有几个模块可以帮助处理这个问题。 - ThisSuitIsBlackNot
2个回答

7

如前所述,您正在寻找笛卡尔积

use strict;
use warnings;

sub getCartesian {
#
  my @input = @_;
  my @ret = map [$_], @{ shift @input };

  for my $a2 (@input) {
    @ret = map {
      my $v = $_;
      map [@$v, $_], @$a2;
    }
    @ret;
  }
  return @ret;
}

my %ranges = (
    '--tic' => [ 0, 1, 2 ],
    '--threads' => [ 8, 16 ],
    '--level' => [ 10, 20 ]
);

my @arr =  map {
  my $k = $_;
  [ map "$k $_", @{$ranges{$k}} ];
}
keys %ranges;

print "@$_\n" for getCartesian(@arr);

输出

--level 10 --tic 0 --threads 8
--level 10 --tic 0 --threads 16
--level 10 --tic 1 --threads 8
--level 10 --tic 1 --threads 16
--level 10 --tic 2 --threads 8
--level 10 --tic 2 --threads 16
--level 20 --tic 0 --threads 8
--level 20 --tic 0 --threads 16
--level 20 --tic 1 --threads 8
--level 20 --tic 1 --threads 16
--level 20 --tic 2 --threads 8
--level 20 --tic 2 --threads 16

4
该模块可以帮助您完成此操作,模块名称为Set::Product
以下是一个示例程序。
use strict;
use warnings 'all';

use Set::Product 'product';

my %ranges = (
    '--tic'     => [ 0, 1, 2 ],
    '--threads' => [ 8, 16 ],
    '--level'   => [ 10, 20 ],
);

my @keys = sort keys %ranges;

product {
    print join(' ', map { "$keys[$_] $_[$_]" } 0 .. $#keys), "\n";
} @ranges{@keys};

输出

--level 10 --threads 8 --tic 0
--level 10 --threads 8 --tic 1
--level 10 --threads 8 --tic 2
--level 10 --threads 16 --tic 0
--level 10 --threads 16 --tic 1
--level 10 --threads 16 --tic 2
--level 20 --threads 8 --tic 0
--level 20 --threads 8 --tic 1
--level 20 --threads 8 --tic 2
--level 20 --threads 16 --tic 0
--level 20 --threads 16 --tic 1
--level 20 --threads 16 --tic 2

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接