in reply to subroutine memory variable scope

You're maybe looking for something like this?

use strict; use warnings; sub analyse_user_inputs { my $hash_ref = shift @_; my @the_inputs; foreach ( keys %{$hash_ref} ) { push @the_inputs, $_; if (ref $hash_ref->{$_} eq 'HASH') { push @the_inputs, @{ analyse_user_inputs($hash_ref->{$_}) +}; } elsif (ref $hash_ref->{$_} eq 'ARRAY') { foreach my $array_element (@{$hash_ref->{$_}}){ push @the_inputs, @{ analyse_user_inputs($array_elemen +t) }; } } } return(\@the_inputs); } my $data = { foo => 1, bar => [ { xyzzy => 'magic', } ], baz => { quux => 123, } }; use Data::Dumper; print Dumper(analyse_user_inputs($data));

No need to play funny tricks with variable scoping. Just take the results of your recursive calls and push them onto @the_inputs.

perl -E'sub Monkey::do{say$_,for@_,do{($monkey=[caller(0)]->[3])=~s{::}{ }and$monkey}}"Monkey say"->Monkey::do'

Replies are listed 'Best First'.
Re^2: subroutine memory variable scope
by austinj (Acolyte) on Dec 13, 2012 at 17:41 UTC

    I think this is close - I made a few changes but I am still only getting the 1st tier of keys when I really want all sublevels of keys. I would like $user_inputs to be an array ref which contains:

    profiles, linewidth, single_dotnotes, legend, xvars, yvars

    when I try pushing the return of the subfunction I get a bunch of wierd empty returns

    function call

    my $user_inputs = analyse_user_inputs($Settings); print Dumper($user_inputs);

    incoming hash/array combo

    $Settings = { batchplot_output_filename => "gps3", plot_all_variables => 'off', # DEFAULT PLOTS & VALUES default_values => [ { profiles => [qw(prof1 prof2)], linewidth => ["2"], single_dotnotes => 'On', legend => ["On"], }, ], #BEGIN CUSTOM PLOTS custom_plots => [ [ { yvars => 'LATDIIP', xvars => 'ELONGI', }, ], ] }

    subfunction

    sub analyse_user_inputs { my $hashORarray_ref = shift @_; my @the_inputs; if (ref $hashORarray_ref eq 'HASH') { foreach ( keys %{$hashORarray_ref} ){ # print "$_ \n"; push @the_inputs , $_ ; analyse_user_inputs($hashORarray_ref->{$_}); } } if (ref $hashORarray_ref eq 'ARRAY') { for (my $i = 0; $i < scalar(@{$hashORarray_ref}); $i++) { analyse_user_inputs($hashORarray_ref->[$i]); } } return(\@the_inputs); }

      "I made a few changes but I am still only getting the 1st tier of keys"

      Then undo the changes, because the code I posted returns all levels of keys.

      The key part which you stripped out is...

      push @the_inputs, @{ analyse_user_inputs(...) };

      When you understand what that's doing, then you'll understand why you shouldn't have stripped it out.

      perl -E'sub Monkey::do{say$_,for@_,do{($monkey=[caller(0)]->[3])=~s{::}{ }and$monkey}}"Monkey say"->Monkey::do'
        The code you posted only works if the first tiers are all hashes - if they are arrays it won't work - which the example I posted has arrays - anyway I switched to checking against a hash since it should be faster processing. The following code works well regardless of how the reference comes in (HASH or ARRAY)
        sub get_user_keys { # subfunction traverses the a hash or array reference for all of the k +eys in all tiers my $hashORarray_ref = shift @_; my %the_inputs; if (ref $hashORarray_ref eq 'HASH') { foreach ( keys %{$hashORarray_ref} ){ # print "$_ \n"; unless(defined $the_inputs{$_}){$the_inputs{$_} = 1} @the_inputs{ keys %{get_user_keys($hashORarray_ref->{$_})} } = v +alues %{get_user_keys($hashORarray_ref->{$_})}; } } if (ref $hashORarray_ref eq 'ARRAY') { for (my $i = 0; $i < scalar(@{$hashORarray_ref}); $i++) { @the_inputs{ keys %{get_user_keys($hashORarray_ref->[$i])} } = v +alues %{get_user_keys($hashORarray_ref->[$i])}; } } return(\%the_inputs); }