Реверсирование пар ключей-значений массива с помощью sed или замены шаблона или раскрытия фигурных скобок?

Реверсирование пар ключей-значений массива с помощью sed или замены шаблона или раскрытия фигурных скобок?
#!/bin/bash
arr=(a b c d)
declare -A brr
for i in ${!arr[@]}
do
  brr[${arr[$i]}]=$i
done
echo ${brr[@]}                             #0 1 2 3

Есть ли способ обратить значения и пары ключей с помощью sedили любой другой команды. Я думаю, мне придется простосделать некоторое изменение шаблона текста вместо использования циклов bash, возможно что-то вроде того.

brr=($(echo ${!arr[@]} | sed commands))

так что новый массив становится таким, как brr=([a]=0 [b]=1 [c]=2 [d]=3) если бы я использовал раскрытие фигурных скобок, он мог бы дойти до этой точки:

#!/bin/bash
arr=(a b c d)
declare -A brr
var=${arr[@]}
var=${var// /,}
brr=$(eval echo [{$var}]=)
echo $brr                              #[a]= [b]= [c]= [d]=

теперь мне просто нужно добавить ключи 0,1,2,3

РЕДАКТИРОВАТЬ: Я смог это сделать, но могу ли я сделать это короче и проще, особенно линии sed?

#!/bin/bash
arr=(a b c d)
declare -A Rev
var=${arr[@]}                    #var is equal to a b c d
var=${var// /,}                  #var is equal to a,b,c,d after adding , for space
brr=$(eval echo Rev[{$var}]=)    #brr equal to Rev[a]= Rev[b]= Rev[c]= Rev[d]=

#NOW I NEED TO CHANGE 
#Rev[a]= Rev[b]= Rev[c]= Rev[d]=
#to
#Rev['a']=0;Rev['b']=1;Rev['c']=2;Rev['d']=3

r="$(echo $brr | 
sed 's/ /\n/g' | sed '/./=' | sed '/./N; s/\n//' | 
  sed 's/\(^[0-9]\{1,\}\)\(.*\)/\2$(echo \1 - 1 | bc)/' | tr '\n' ';' |
                                                          sed "s/\[/\[\'/g;s/\]/\'\]/g")"
#pipe the output of echo(Rev[a]= Rev[b]= Rev[c]= Rev[d]=) to sed
#convert space to newlines
#add line numbers
#put the line numbers adjacent to the values Rev[a],Rev[b],Rev[c],Rev[d] like
# 1rev[a]=
# 2Rev[b]=
# 3Rev[c]=
# 4Rev[d]=
#change 1Rev[a]= format to Rev[a]=1
#convert newlines to ; looking like Rev[a]=0;Rev[b]=1;Rev[c]=2;Rev[d]=3
#put quotes around the keys Rev['a']=0;Rev['b']=1;Rev['c']=2;Rev['d']=3

eval $r
echo ${Rev[@]}      #0 1 2 3
echo ${!Rev[@]}     #a b c d

решение1

Я бы прислушался к совету @ChrisDown и не пытался бы здесь что-то умное сделать. Я бы создал цикл for и поменял бы ключи со значениями в новом массиве. Это всего несколько строк кода.

Любое созданное вами таким образом решение будет сложнее поддерживать другим в будущем, а вам будет сложнее запомнить/понять, как оно работает, когда вы в будущем будете возвращаться к коду.

решение2

#!/bin/bash
arr=(a b c d)
set -- $(echo ${arr[@]} | tr ' ' '\n' | sed "s/.*/arr['&']=/")
eval echo $(seq 1 ${#arr[@]} | sed 's/\(.*\)/${\1}$(echo \1 -1 | bc)/') 

или

#!/bin/bash
declare -A brr
arr=(a b c d)
eval "brr=($(eval echo $(sed 's/\([0-9]\)/[${arr[\1]}]=\1/g' <(echo ${!arr[@]}))))"
echo ${brr[@]}

Выход:

#arr['a']=0 arr['b']=1 arr['c']=2 arr['d']=3

решение3

Как уже говорили другие, я думаю, sedэто будет трудным решением.

Следующий цикл может достичь желаемого:

arr=(a b c d)
for i in ${!arr[@]}; do
    brr+=("[${arr[$i]}]=$i")
done
echo ${brr[@]}
#[a]=0 [b]=1 [c]=2 [d]=3

Связанный контент