在 Perl 中从数组中删除值的最佳方法是啥?

Posted

技术标签:

【中文标题】在 Perl 中从数组中删除值的最佳方法是啥?【英文标题】:What is the best way to delete a value from an array in Perl?在 Perl 中从数组中删除值的最佳方法是什么? 【发布时间】:2010-09-15 12:21:37 【问题描述】:

数组有很多数据,我需要删除两个元素。

下面是我正在使用的代码sn-p,

my @array = (1,2,3,4,5,5,6,5,4,9);
my $element_omitted = 5;
@array = grep  $_ != $element_omitted  @array;

【问题讨论】:

这会删除三个元素。 needed top remove all non-file item form directory lists and "array = grep -f $_ array" 对我来说就像一个魅力:) 【参考方案1】:

如果您已经知道要删除的元素的索引,请使用拼接。

如果您正在搜索,Grep 会起作用。

如果你需要做很多这样的事情,如果你让你的数组保持排序,你会得到更好的性能,因为你可以进行二进制搜索来找到必要的索引。

如果在您的上下文中有意义,您可能需要考虑对已删除记录使用“神奇值”,而不是删除它们,以节省数据移动 - 例如,将已删除元素设置为 undef。当然,这有其自身的问题(如果您需要知道“活动”元素的数量,则需要单独跟踪它等),但根据您的应用程序可能值得麻烦。

编辑实际上,现在我再看一遍——不要使用上面的 grep 代码。找到要删除的元素的索引,然后使用splice删除它会更有效(你的代码累积了所有不匹配的结果..)

my $index = 0;
$index++ until $arr[$index] eq 'foo';
splice(@arr, $index, 1);

这将删除第一个匹配项。 删除所有匹配项非常相似,只是您希望一次性获取所有索引:

my @del_indexes = grep  $arr[$_] eq 'foo'  0..$#arr;

剩下的留给读者练习——记住数组在拼接时会发生变化!

Edit2 John Siracusa 正确地指出我的示例中有一个错误。已修复,对此感到抱歉。

【讨论】:

如果找不到字符串,循环就会卡住,我的$index = 0;也是这样我的 $count = 标量 @arr; $index++ 直到 $arr[$index] eq 'foo' 或 $index==$count;拼接(@arr, $index, 1); my ($index) = grep $arr[$_] eq 'foo' 0..$#arr; if (defined $index) splice(@arr, $index, 1); - 第一场比赛【参考方案2】:

splice 将按索引删除数组元素。如您的示例所示,使用 grep 进行搜索和删除。

【讨论】:

谢谢斯波尔森。我不知道我必须删除的索引,所以我不得不求助于 grep。【参考方案3】:

这是你会经常做的事情吗?如果是这样,您可能需要考虑不同的数据结构。 Grep 每次都会搜索整个数组,对于一个大数组可能会非常昂贵。如果速度是一个问题,那么您可能需要考虑使用 Hash。

在您的示例中,键是数字,值是该数字的元素计数。

【讨论】:

【参考方案4】:

如果你改变了

my @del_indexes = grep  $arr[$_] eq 'foo'  0..$#arr;

my @del_indexes = reverse(grep  $arr[$_] eq 'foo'  0..$#arr);

这通过首先从数组后面删除元素来避免数组重新编号问题。 将 splice() 放入 foreach 循环中会清理 @arr。比较简单易读...

foreach $item (@del_indexes) 
   splice (@arr,$item,1);

【讨论】:

【参考方案5】:

您可以使用数组切片而不是拼接。 Grep 返回您想要保留的索引并使用切片:

my @arr = ...;
# run through each item.
my @indicesToKeep = grep  $arr[$_] ne 'foo'  0..$#arr;
@arr = @arr[@indicesToKeep];

【讨论】:

我特别喜欢这种方法的逻辑性和优雅性。 是的,您甚至可以将它写成一个单行字,例如:@arr = @arr[grep ...],我特别喜欢。我不确定它的效率如何,但我会开始使用它,因为它不会比其他解决方案差。【参考方案6】:

我发现最好的是“undef”和“grep”的组合:

foreach $index ( @list_of_indexes_to_be_skiped ) 
      undef($array[$index]);

@array = grep  defined($_)  @array;

这就是诀窍! 费德里科

【讨论】:

undef 将元素值设置为 null。总元素(大小)仍然相同。 @BoontaweeHome,最后的grep 然后删除它们。 如果您已经知道索引,只需使用splice【参考方案7】:

你可以这样做:

my $input_Color = 'Green';
my @array = qw(Red Blue Green Yellow Black);
@array = grep !/$input_Color/ @array;
print "@array";

【讨论】:

【参考方案8】:

我认为您的解决方案是最简单且最易于维护的。

帖子的其余部分记录了将元素测试转换为splice 偏移量的难度。因此,使它成为一个更完整的答案。

查看 gyrations,您必须通过有效(即一次性)算法将列表项的测试转换为索引。而且它根本不是那么直观。

sub array_remove ( \@& )  
    my ( $arr_ref, $test_block ) = @_;
    my $sp_start  = 0;
    my $sp_len    = 0;
    for ( my $inx = 0; $inx <= $#$arr_ref; $inx++ ) 
        local $_ = $arr_ref->[$inx];
        next unless $test_block->( $_ );
        if ( $sp_len > 0 && $inx > $sp_start + $sp_len ) 
            splice( @$arr_ref, $sp_start, $sp_len );
            $inx    = $inx - $sp_len;
            $sp_len = 0;
        
        $sp_start = $inx if ++$sp_len == 1;
    
    splice( @$arr_ref, $sp_start, $sp_len ) if $sp_len > 0;
    return;

【讨论】:

一个简单的“grep”将比这更容易理解和更有效。 有人删除了我的评论,说你显然没有看文字。【参考方案9】:

我用:

delete $array[$index];

Perldoc delete.

【讨论】:

delete 数组值可能已被弃用(请参阅您的文档) 这只是删除存储在该数组索引处的值。至少在我的 perl 版本中,(5.14) 这并没有真正删除你的想法。它只删除该值,使其成为undef。此外,来自 ringø 链接的文档:“警告:强烈建议不要对数组值调用 delete。删除或检查 Perl 数组元素的存在的概念在概念上并不连贯,并且可能导致令人惊讶的行为。” (文档中的前一段包含所有血淋淋的细节)。【参考方案10】:

删除所有出现的 'something' if 数组。

基于 SquareCog 的回答:

my @arr = ('1','2','3','4','3','2', '3','4','3');
my @dix = grep  $arr[$_] eq '4'  0..$#arr;
my $o = 0;
for (@dix) 
    splice(@arr, $_-$o, 1);
    $o++;

print join("\n", @arr);

每次我们从@arr 中删除索引时,下一个要删除的正确索引将是$_-current_loop_step

【讨论】:

【参考方案11】:

您可以使用非捕获组和要删除的项目的管道分隔列表。


perl -le '@ar=(1 .. 20);@x=(8,10,3,17);$x=join("|",@x);@ar=grep!/^(?:$x)$/o @ar;print "@ar"'

【讨论】:

【参考方案12】:

只是为了确保我已经对 grep 和 map 解决方案进行了基准测试,首先搜索匹配元素的索引(要删除的元素),然后通过 grep 直接删除元素而不搜索索引。 看来山姆在问他问题时提出的第一个解决方案已经是最快的了。

    use Benchmark;
    my @A=qw(A B C A D E A F G H A I J K L A M N);
    my @M1; my @G; my @M2;
    my @Ashrunk;
    timethese( 1000000, 
      'map1' => sub 
          my $i=0;
          @M1 = map  $i++; $_ eq 'A' ? $i-1 : (); @A;
      ,
      'map2' => sub 
          my $i=0;
          @M2 = map  $A[$_] eq 'A' ? $_ : () ; 0..$#A;
      ,
      'grep' => sub 
          @G = grep  $A[$_] eq 'A'  0..$#A;
      ,
      'grem' => sub 
          @Ashrunk = grep  $_ ne 'A'  @A;
      ,
    );

结果是:

Benchmark: timing 1000000 iterations of grem, grep, map1, map2...
  grem:  4 wallclock secs ( 3.37 usr +  0.00 sys =  3.37 CPU) @ 296823.98/s (n=1000000)
  grep:  3 wallclock secs ( 2.95 usr +  0.00 sys =  2.95 CPU) @ 339213.03/s (n=1000000)
  map1:  4 wallclock secs ( 4.01 usr +  0.00 sys =  4.01 CPU) @ 249438.76/s (n=1000000)
  map2:  2 wallclock secs ( 3.67 usr +  0.00 sys =  3.67 CPU) @ 272702.48/s (n=1000000)
M1 = 0 3 6 10 15
M2 = 0 3 6 10 15
G = 0 3 6 10 15
Ashrunk = B C D E F G H I J K L M N

从经过的时间可以看出,尝试实现remove是没有用的 函数使用 grep 或 map 定义的索引。直接grep-remove即可。

在测试之前,我认为“map1”将是最有效的......我猜我应该更频繁地依赖 Benchmark。 ;-)

【讨论】:

【参考方案13】:

如果你知道数组索引,你可以delete()它。 splice() 和 delete() 的区别在于 delete() 不会对数组的剩余元素重新编号。

【讨论】:

我实际上的意思是重新编号,根据 Perldoc,splice() 确实如此。【参考方案14】:

我曾经写过一个类似的代码,用于从字符串数组中删除不以 SB.1 开头的字符串

my @adoSymbols=('SB.1000','RT.10000','PC.10000');
##Remove items from an array from backward
for(my $i=$#adoSymbols;$i>=0;$i--)   
    unless ($adoSymbols[$i] =~ m/^SB\.1/) splice(@adoSymbols,$i,1);

【讨论】:

【参考方案15】:

这也很好用:

my @array = (1,2,3,4,5,5,6,5,4,9);
my $element_omitted = 5;
for( my $i = 0; $i < scalar( @array ); $i++ )

    splice( @array, $i ), $i-- if( $array[$i] == $element_omitted );

say "@array"; # 1 2 3 4 6 4 9

【讨论】:

以上是关于在 Perl 中从数组中删除值的最佳方法是啥?的主要内容,如果未能解决你的问题,请参考以下文章

在部署后脚本中使用已删除列中的值的最佳方法是啥

在 Perl 中处理异常的最佳方法是啥?

在 PHP 中删除数组项的最佳方法是啥?

淘汰赛,在 observableArray 中找到值的最佳方法是啥

在 Perl 中获得 Epoch 毫秒的最佳方法是啥?

在 Perl 中打开和读取文件的最佳方法是啥?