Home > front end >  Bash - How to assign list to array member
Bash - How to assign list to array member

Time:02-24

In bash, I've an array with values:

declare -A my_array
my_array['key1']='value1'
my_array['key2']='value2'

echo "-----my_array-----"
echo key1: ${my_array['key1']}
echo key2: ${my_array['key2']}

what works well:

-----my_array----- 
key1: value1 
key2: value2

Although, I'd like to have for a key, a list of values. I've tried something like:

my_array['key2']=('value2.1', 'value2.2')  

this gives the error:

my_array['key2']: cannot assign list to array member

What is the Syntax to add a list into an array element and then to access an index values for a key?

CodePudding user response:

It's not possible directly but there is a way to do this via variable expansion, like this:

declare -A my_array
my_array['key1']='value1'
my_array['key2']='value2'

value1="true value1"
value2="true value2"

Check:

$ echo ${!my_array['key1']}
true value1

$ echo ${!my_array['key2']}
true value2

It's more complicated if value* is an array, but also possible:

...
value1=("true value1.1" "true value1.2")
value1=("true value2.1" "true value2.2")
index="${my_array['key1']}[0]"

$ echo "${!index}"
true value2.1

This could be wrapped in a function:

fun(){
    local index1=${!1}
    local index2="$index1[$2]"
    echo  "${!index2}"
}

$ fun "my_array['key2']" 1
true value2.2

But IMO you should accept the limitation and rethink the design or consider to use another language.

CodePudding user response:

No. Thechnically, shell variables are only intended to hold strings. But...

Creating array of arrays (even associative), under

I will use disk free: df -k output to create associatives array, indexed by device. As tmpfs pseudo device is used for many differents behaviour, this field must contain a list.

Simpliest way, using special delimiter.

Here I use spaces to delimit contents of fields as they are space delimited by command. Then I use bell special character (\007) to delimit sub arrays.

Here is a sample, using df output, sorting creating lists by device. As result, all tmpfs will be stored together:

declare -A dfVars
{
    read _;
    while read -r dev _ use free _ mpnt ;do
        dfVars["${dev##*/}"] ="$use $((use free)) $mpnt"$'\07'
    done
} < <(
    LANG=C df -lk
)

You could parse:

for dev in ${!dfVars[@]} ;do
    IFS=$'\a\n' read -d '' -ra fields <<<"${dfVars["$dev"]%$'\a'}"
    for line in "${fields[@]}";do
        read -r use tot mpnt <<<"$line"
        printf "%-20s d d  %s\n" $dev $use $tot $mpnt
    done
done

May output someting like:

tmpfs                          0      188928  /dev/shm
tmpfs                      19284      188928  /run
tmpfs                          4        5120  /run/lock
tmpfs                          0      188928  /sys/fs/cgroup
tmpfs                          0       37784  /run/user/33
tmpfs                          0       37784  /run/user/1001
devtmpfs                       0      184596  /dev
mmcblk0p1                  23193       42136  /boot
root                   115253500   117652696  /

Everything are stored into $dfVars:

declare -p dfVars
declare -A dfVars=([tmpfs]=$'0 188928 /dev/shm\a19284 188928 /run\a4 5120 /run/l
ock\a0 188928 /sys/fs/cgroup\a0 37784 /run/user/33\a0 37784 /run/user/1001\a' [d
evtmpfs]=$'0 184596 /dev\a' [mmcblk0p1]=$'23193 42136 /boot\a' [root]=$'11529034
8 117652696 /\a' )

Stronger: Using nameref for indexing arrays:

With this method, you create independant variables, wich could be array, associative array or even strings...

uuvar() { printf -v "$1" %s	d "$2" $((  uuVarCnt));}
declare -A dfVars
{
     read _
     while read -r dev _ use free _ mpnt ;do
         uuvar vname _df_
         dfVars["${dev##*/}"] =$vname' ' 
         read -a $vname <<<"$use $((use free)) $mpnt"
         done
} < <(LANG=C df -lk)

To parse this:

for dev in ${!dfVars[@]} ;do
    for sub in ${dfVars["$dev"]};do
        declare -n subDf=$sub
        printf "%-20s d d  %s\n" "$dev" "${subDf[@]}"
    done
done
tmpfs                          0      188928  /dev/shm
tmpfs                      19284      188928  /run
tmpfs                          4        5120  /run/lock
tmpfs                          0      188928  /sys/fs/cgroup
tmpfs                          0       37784  /run/user/33
tmpfs                          0       37784  /run/user/1001
devtmpfs                       0      184596  /dev
mmcblk0p1                  23193       42136  /boot
root                   115254628   117652696  /

Where $dfVars and sub variables look like:

declare -p dfVars uuVarCnt;set | grep ^_df_
declare -A dfVars=([tmpfs]="_df_000000003 _df_000000004 _df_000000005 _df_000000
006 _df_000000008 _df_000000009 " [devtmpfs]="_df_000000002 " [mmcblk0p1]="_df_0
00000007 " [root]="_df_000000001 " )
declare -- uuVarCnt="9"
_df_000000001=([0]="115335328" [1]="117652696" [2]="/")
_df_000000002=([0]="0" [1]="184596" [2]="/dev")
_df_000000003=([0]="0" [1]="188928" [2]="/dev/shm")
_df_000000004=([0]="19284" [1]="188928" [2]="/run")
_df_000000005=([0]="4" [1]="5120" [2]="/run/lock")
_df_000000006=([0]="0" [1]="188928" [2]="/sys/fs/cgroup")
_df_000000007=([0]="23193" [1]="42136" [2]="/boot")
_df_000000008=([0]="0" [1]="37784" [2]="/run/user/33")
_df_000000009=([0]="0" [1]="37784" [2]="/run/user/1001")

Same, but with associative arrays:

declare -A dfVars
uuvar() { printf -v "$1" %s	d "$2" $((  uuVarCnt));}
{
    read _
    while read -r dev _ use free _ mpnt ;do
        uuvar vname _df_
        dfVars["${dev##*/}"] =$vname' '
        declare -A $vname
        printf -v $vname[used] %d $use
        printf -v $vname[total] %d $((use free))
        printf -v $vname[mount\ point] %s "$mpnt"
    done
} < <(LANG=C df -lk)
for dev in ${!dfVars[@]} ;do
    for sub in ${dfVars["$dev"]} ;do
        declare -n subDf=$sub
        printf "%-20s d d  %s\n" "$dev" \
            "${subDf[used]}" "${subDf[total]}" "${subDf["mount point"]}"
    done
done
## < snip same output >

Then

declare -p dfVars uuVarCnt;set | grep ^_df_
declare -A dfVars=([tmpfs]="_df_000000003 _df_000000004 _df_000000005 _df_000000
006 _df_000000008 _df_000000009 " [devtmpfs]="_df_000000002 " [mmcblk0p1]="_df_0
00000007 " [root]="_df_000000001 " )
declare -- uuVarCnt="9"
_df_000000001=([used]="115358360" [total]="117652696" ["mount point"]="/" )
_df_000000002=([used]="0" [total]="184596" ["mount point"]="/dev" )
_df_000000003=([used]="0" [total]="188928" ["mount point"]="/dev/shm" )
_df_000000004=([used]="19284" [total]="188928" ["mount point"]="/run" )
_df_000000005=([used]="4" [total]="5120" ["mount point"]="/run/lock" )
_df_000000006=([used]="0" [total]="188928" ["mount point"]="/sys/fs/cgroup" )
_df_000000007=([used]="23193" [total]="42136" ["mount point"]="/boot" )
_df_000000008=([used]="0" [total]="37784" ["mount point"]="/run/user/33" )
_df_000000009=([used]="0" [total]="37784" ["mount point"]="/run/user/1001" )
  • Related