Search code examples
arraysbashglobal-variablesparameter-passing

How to return an array in bash without using globals?


I have a function that creates an array and I want to return the array to the caller:

create_array() {
  local my_list=("a", "b", "c")
  echo "${my_list[@]}"
}

my_algorithm() {
  local result=$(create_array)
}

With this, I only get an expanded string. How can I "return" my_list without using anything global?


Solution

  • What's wrong with globals?

    Returning arrays is really not practical. There are lots of pitfalls.

    That said, here's one technique that works if it's OK that the variable have the same name:

    $ f () { local a; a=(abc 'def ghi' jkl); declare -p a; }
    $ g () { local a; eval $(f); declare -p a; }
    $ f; declare -p a; echo; g; declare -p a
    declare -a a='([0]="abc" [1]="def ghi" [2]="jkl")'
    -bash: declare: a: not found
    
    declare -a a='([0]="abc" [1]="def ghi" [2]="jkl")'
    -bash: declare: a: not found
    

    The declare -p commands (except for the one in f() are used to display the state of the array for demonstration purposes. In f() it's used as the mechanism to return the array.

    If you need the array to have a different name, you can do something like this:

    $ g () { local b r; r=$(f); r="declare -a b=${r#*=}"; eval "$r"; declare -p a; declare -p b; }
    $ f; declare -p a; echo; g; declare -p a
    declare -a a='([0]="abc" [1]="def ghi" [2]="jkl")'
    -bash: declare: a: not found
    
    -bash: declare: a: not found
    declare -a b='([0]="abc" [1]="def ghi" [2]="jkl")'
    -bash: declare: a: not found