Is perl's each function worth using?

后端 未结 8 1673
忘掉有多难
忘掉有多难 2021-01-17 11:37

From perldoc -f each we read:

There is a single iterator for each hash, shared by all each, keys, and values

相关标签:
8条回答
  • 2021-01-17 12:07

    use the keys() function to reset the iterator. See the faq for more info

    0 讨论(0)
  • 2021-01-17 12:09

    each is not only worth using, it's pretty much mandatory if you want to loop over all of a tied hash too big for memory.

    A void-context keys() (or values, but consistency is nice) before beginning the loop is the only "workaround" necessary; is there some reason you are looking for some other workaround?

    0 讨论(0)
  • 2021-01-17 12:16

    I think it is worth using as long as you are aware of this. It's ideal when you need both key and value in iteration:

    while (my ($k,$v) = each %h) {
        say "$k = $v";
    }
    

    In your example you can reset the iterator by adding keys %h; like so:

    my %h = map { $_ => 1 } qw/1 2 3/;
    while (my $k = each %h) { print "1: $k\n"; last }
    keys %h;  # reset %h
    while (my $k = each %h) { print "2: $k\n" }
    

    From Perl 5.12 each will also allow iteration on an array.

    0 讨论(0)
  • 2021-01-17 12:16

    It's best if used as it's name: each. It's probably the wrong thing to use if you mean "give me the first key-value pair," or "give me the first two pairs" or whatever. Just keep in mind that the idea is flexible enough that each time you call it, you get the next pair (or key in a scalar context).

    0 讨论(0)
  • 2021-01-17 12:19

    each() can be more efficient if you are iterating through a tied hash, for example a database that contains millions of keys; that way you don't have to load all the keys in memory.

    0 讨论(0)
  • 2021-01-17 12:26

    each has a buit-in, hidden global variable that can hurt you. Unless you need this behavior, it's safer to just use keys.

    Consider this example where we want to group our k/v pairs (yes, I know printf would do this better):

    #!perl
    
    use strict;
    use warnings;
    
    use Test::More 'no_plan';
    
    {   my %foo = map { ($_) x 2 } (1..15);
    
        is( one( \%foo ), one( \%foo ), 'Calling one twice works with 15 keys' );
        is( two( \%foo ), two( \%foo ), 'Calling two twice works with 15 keys' );
    }
    
    {   my %foo = map { ($_) x 2 } (1..105);
    
        is( one( \%foo ), one( \%foo ), 'Calling one twice works with 105 keys' );
        is( two( \%foo ), two( \%foo ), 'Calling two twice works with 105 keys' );
    }
    
    
    sub one {
        my $foo = shift;
    
        my $r = '';
    
        for( 1..9 ) {
            last unless my ($k, $v) = each %$foo;
    
            $r .= "  $_: $k -> $v\n";
        }
        for( 10..99 ) {
            last unless my ($k, $v) = each %$foo;
    
            $r .= " $_: $k -> $v\n";
        }
    
        return $r;
    }
    
    sub two {
        my $foo = shift;
    
        my $r = '';
    
        my @k = keys %$foo;
    
        for( 1..9 ) {
            last unless @k;
            my $k = shift @k;
    
            $r .= "  $_: $k -> $foo->{$k}\n";
        }
        for( 10..99 ) {
            last unless @k;
            my $k = shift @k;
    
            $r .= "  $_: $k -> $foo->{$k}\n";
        }
    
        return $r;
    }
    

    Debugging the error shown in the tests above in a real application would be horribly painful. (For better output use Test::Differences eq_or_diff instead of is.)

    Of course one() can be fixed by using keys to clear the iterator at the start and end of the subroutine. If you remember. If all your coworkers remember. It's perfectly safe as long as no one forgets.

    I don't know about you, but I'll just stick with using keys and values.

    0 讨论(0)
提交回复
热议问题