#!/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