我有两个字符串数组,我想比较相等:
my @array1 = ("part1","part2","part3","part4"); my @array2 = ("part1","PART2","part4");
是否有一个内置的方法来比较数组,就像有标量?
我试过了:
if (@array1 == @array2) {...}
但它只是在标量上下文中评估每个数组,因此比较每个数组的长度。
我可以滚动我自己的函数来做它,但它似乎像这样一个低级操作,应该有一个内置的方法来做到这一点。在那儿?
编辑:可悲的是,我没有访问5.10或可选组件。
解决方法
有新的
smart match operator:
#!/usr/bin/perl use 5.010; use strict; use warnings; my @x = (1,2,3); my @y = qw(1 2 3); say "[@x] and [@y] match" if @x ~~ @y;
Internally the comparator compares the two arrays by using join to turn both arrays into strings and comparing the strings using
eq
.
我想这是一个有效的方法,但只要我们使用字符串比较,我宁愿使用像:
#!/usr/bin/perl use strict; use warnings; use List::AllUtils qw( each_arrayref ); my @x = qw(1 2 3); my @y = (1,3); print "[@x] and [@y] match\n" if elementwise_eq( \(@x,@y) ); sub elementwise_eq { my ($xref,$yref) = @_; return unless @$xref == @$yref; my $it = each_arrayref($xref,$yref); while ( my ($x,$y) = $it->() ) { return unless $x eq $y; } return 1; }
如果你正在比较的数组很大,加入它们将做大量的工作,并消耗大量的内存,而不仅仅是逐个比较每个元素。
更新:当然,应该测试这样的语句。简单基准:
#!/usr/bin/perl use strict; use warnings; use Array::Compare; use Benchmark qw( cmpthese ); use List::AllUtils qw( each_arrayref ); my @x = 1 .. 1_000; my @y = map { "$_" } 1 .. 1_000; my $comp = Array::Compare->new; cmpthese -5,{ iterator => sub { my $r = elementwise_eq(\(@x,@y)) },array_comp => sub { my $r = $comp->compare(\(@x,};
这是最糟糕的情况,其中elementwise_eq必须遍历两个数组中的每个元素1_000次,它显示:
Rate iterator array_comp iterator 246/s -- -75% array_comp 1002/s 308% --
另一方面,最好的情况是:
my @x = map { rand } 1 .. 1_000; my @y = map { rand } 1 .. 1_000;
Rate array_comp iterator array_comp 919/s -- -98% iterator 52600/s 5622% --
迭代器性能下降很快,但是:
my @x = 1 .. 20,map { rand } 1 .. 1_000; my @y = 1 .. 20,map { rand } 1 .. 1_000;
Rate iterator array_comp iterator 10014/s -- -23% array_comp 13071/s 31% --
我没有看内存利用率。