В Perl, как я могу перебирать несколько элементов массива?

у меня есть файл CSV, который я использую split для разбора в массив N элементов, где N несколько 3.

есть ли способ сделать это

foreach my ( $a, $b, $c ) ( @d ) {}

похоже на Python?

5 ответов


можно использовать список:: MoreUtils:: natatime. Из документов:

my @x = ('a' .. 'g');
my $it = natatime 3, @x;
while (my @vals = $it->()) {
    print "@vals\n";
}

natatime реализован в XS, поэтому вы должны предпочесть его для эффективности. Просто для иллюстрации, вот как можно реализовать генератор итераторов трех элементов в Perl:

#!/usr/bin/perl

use strict; use warnings;

my @v = ('a' .. 'z' );

my $it = make_3it(\@v);

while ( my @tuple = $it->() ) {
    print "@tuple\n";
}

sub make_3it {
    my ($arr) = @_;
    {
        my $lower = 0;
        return sub {
            return unless $lower < @$arr;
            my $upper = $lower + 2;
            @$arr > $upper or $upper = $#$arr;
            my @ret = @$arr[$lower .. $upper];
            $lower = $upper + 1;
            return @ret;
        }
    }
}

я решил эту проблему в своем модуле Список:: Gen на CPAN.

use List::Gen qw/by/;

for my $items (by 3 => @list) {

    # do something with @$items which will contain 3 element slices of @list

    # unlike natatime or other common solutions, the elements in @$items are
    # aliased to @list, just like in a normal foreach loop

}

вы также можете импортировать mapn функция, которая используется List::Gen для реализации by:

use List::Gen qw/mapn/;

mapn {

   # do something with the slices in @_

} 3 => @list;

@z=(1,2,3,4,5,6,7,8,9,0);

for( @tuple=splice(@z,0,3); @tuple; @tuple=splice(@z,0,3) ) 
{ 
  print "$tuple[0] $tuple[1] $tuple[2]\n"; 
}

выдает:

1 2 3
4 5 6
7 8 9
0

my @list = (qw(one two three four five six seven eight nine));

while (my ($m, $n, $o) = splice (@list,0,3)) {
  print "$m $n $o\n";
}

вот результаты:

one two three
four five six
seven eight nine

не легко. Тебе лучше сделать @d массив трехэлементных кортежей, путем нажатия элементов на массив в качестве ссылки на массив:

foreach my $line (<>)
    push @d, [ split /,/, $line ];

(за исключением того, что вы действительно должны использовать один из модулей CSV из CPAN.