Handling missing references

后端 未结 2 1584
深忆病人
深忆病人 2021-01-24 12:23

I\'m getting the error Can\'t use an undefined value as an ARRAY reference in a Perl script. Below is a highly simplified version.

Basically I have setup a

2条回答
  •  不知归路
    2021-01-24 12:36

    So you need to check if $data_for{$letter} is a reference or not. Possible checks:

    • exists($data_for{$letter}).
    • defined($data_for{$letter}) because a non-existent hash element is undefined.
    • $data_for{$letter} because references are always true.

    We can use the check to provide us something to dereference:

    sort @{ $data_for{$letter} // [] }
    

    We can use the check to avoid dereferencing at all.

    sort $data_for{$letter} ? @{ $data_for{$letter} } : ()
    

    This gives us the following:

    print "$letter: ";
    for my $item ( sort $data_for{$letter} ? @{ $data_for{$letter} } : () ) {
        print "$item, ";
    }
    print "\n";
    

    Or better yet, we can avoid the trailing , with the following:

    use feature qw( say );
    
    say "$letter: ", join ", ", sort $data_for{$letter} ? @{ $data_for{$letter} } : ();
    

    It works just fine if I DON'T sort the data array.

    When you dereference an undefined variable that's used as a modifiable value (lvalue), Perl will automatically create a variable of the appropriate type, and it will place a reference to the newly created variable in the undefined variable. This is called "autovivification".

    Since autovivifying only happens in lvalue context, derefencing an undefined value elsewhere leads to the an error.

     my ($a,$b);  # Both are undefined.
     @$a = @b;    # OK. Equivalent to @{ $a //= [] } = @b.
     @a = @$b;    # XXX. "Can't use an undefined value as an ARRAY reference."
    

    sort doesn't modify its operands, so it doesn't evaluate them in lvalue context.

    A foreach loop evaluates its operands in lvalue context to allow for for (@$a) { $_ = uc($_); }. That means that

    for (@{ $data_for{$letter} }) { ... }
    

    implicitly modifies $data_for{$letter} as follows:

    for (@{ $data_for{$letter} //= [] }) { ... }
    

提交回复
热议问题