从哈希值生成组合

时间:2016-04-15 05:22:53

标签: perl hash

在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,但可以是任意数字。

我认为拼接,地图和foreach的某些组合应该可以工作,但我正在以一种糟糕的方式找到它。

2 个答案:

答案 0 :(得分:7)

如前所述,您正在寻找Cartesian product

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

答案 1 :(得分: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