2012-06-09 23 views
6

yılında> {c} Böyle dinamik iç içe karma-ref ettik:

my $hash = { 'a' => { 'b' => { 'c' => 'value' } } }; 

Ben ayarlamak istediğiniz Kullanıcının "abc bir şey" girmesine izin vererek c'nin 'bir şey' değerini.

my $keys = 'a.b.c'; 
my $v='something'; 
my $h = $hash; 
foreach my $k(split /\./, $keys) { 
    $h = $h->{$k}; 
} 
print $h; # "value" 

Ama nasıl

print Dumper $hash; 

değişikliği yansıtır böylece $v için anahtar c değerini ayarlarsınız:

Şimdi alma değeri böyle yapılabilir ? $h, foreach döngüsünün sonunda bir ref değil, bu nedenle değişiklik, $hash'daki değişikliği yansıtmıyor. Kafamdaki düğümleri nasıl çözeceğime dair ipuçları var mı? Böyle

+2

standart 'Veriler :: Dumper' en' Dumper' yerine CPAN 'Veriler :: Dump' modülünün' dd' işlevini kullanmayı deneyin. CPAN modülü daha kolay okumayı sağlar. – tchrist

cevap

7

şey:

my $h = $hash; 
my @split_key = split /\./, $keys; 
my $last_key = pop @split_key; 
foreach my $k (@split_key) { 
    $h = $h->{$k}; 
} 
$h->{$last_key} = $v; 
+0

Teşekkürler, çalışıyor :) – agranig

3
use strictures; 
use Data::Diver qw(DiveVal); 

my ($hash, $path, $value) = (
    { 'a' => { 'b' => { 'c' => 'value' } } }, 
    'a.b.c', 
    'something', 
); 

DiveVal($hash, split /[.]/, $path) = $value; 
# { a => { b => { c => 'something' } } } 
4

dışarı haşhaş son anahtar wusses göre! ;) Bu fonksiyon

sub dive :lvalue { 
    my $r = \shift; 
    $r = \(($$r)->{$_}) for @_; 
    return $$r; 
} 

my $data; 
my $key = 'a.b.c'; 
my $val = 'value'; 

dive($data, split /\./, $key) = $val; 

A) daha güçlü (ve bu nedenle biraz daha sert kullanma sürüm Data::Diver tarafından sağlanır.

use Data::Diver qw(DiveVal); 

my $data; 
my $key = 'a.b.c'; 
my $val = 'value'; 

DiveVal($data //= {}, map \$_, split /\./, $key) = $val; 

(daxim en kullanımı biraz kapalıdır.)