Search code examples
perlhashreturn-valuesubroutine

Perl: Hash values change after returning from a subroutine


I'm working with hashes in perl, but don't understand why the hash value changes in the following:

use strict;

sub test
{
    my ($value_ref) = @_;
    my %value = %$value_ref;
    $value{'abc'}{'xyz'} = 1;
}

my %hash;
$hash{'abc'}{'xyz'} = 0;
test (\%hash);
print "$hash{'abc'}{'xyz'}\n";

The above returns 1, why doesn't it return 0 like this it does here?

use strict;

sub test
{
    my ($value_ref) = @_;
    my %value = %$value_ref;
    $value{'abc'} = 1;
}

my %hash;
$hash{'abc'} = 0;
test (\%hash);
print "$hash{'abc'}\n";

I imagine it has to do with how I'm passing in %hash. What am I missing?


Solution

  • You are getting incorrect results because when you make your copy of the hash using my %value = %$value_ref; you are only getting the top-level keys. The second level, with the key 'xyz', is the one where value is actually stored so any changes to that reference are carried over. You are performing a shallow copy when you need a deep copy.

    Luckily there is a CPAN module for that!

    use strict;
    use Storable qw(dclone);
    
    sub test
    {
        my ($value_ref) = @_;
        my %value = %{ dclone($value_ref) };
        $value{'abc'}{'xyz'} = 1;
    }
    
    my %hash;
    $hash{'abc'}{'xyz'} = 0;
    test (\%hash);
    print "$hash{'abc'}{'xyz'}\n"; # prints 0