引用!!

引用!!

我正在编写一个简单的 bash 脚本,该脚本修复了 Retropie 上的重复命名问题。

该脚本本身会获取 gameslist.xml 文件中多次提及的任何名称,然后将它们存储在数组中以供以后使用。

我最终在索引中循环这个数组,如下所示:

pi@retropie:~ $ for game in ${game_array[@]:0:10} ; do echo $game; done

它拉出第 1 个到第 10 个元素(即${game_array[9]}),但是输出被连接到一行:

pi@retropie:~ $ for game in ${game_array[@]:0:10} ; do echo $game; done
R.B.I. Baseball '94 World Series Baseball '95 Mega Games 1 Bill Walsh College Football T2: The Arcade Game Sonic & Knuckles + Sonic the Hedgehog Sega Top Five Pyramid Magic Tecmo Super Baseball Super Chinese Tycoon

但如果我循环整个数组,它会按预期输出新行:

pi@retropie:~ $ for game in ${game_array[@]}; do echo $game; done | head -10
R.B.I. Baseball '94
World Series Baseball '95
Mega Games 1
Bill Walsh College Football
T2: The Arcade Game
Sonic & Knuckles + Sonic the Hedgehog
Sega Top Five
Pyramid Magic
Tecmo Super Baseball
Super Chinese Tycoon

字段分隔符已设置为新行IFS='$\n',这就是第二个有效的原因,但我一生都无法弄清楚为什么第一个无效?

这是上下文的完整测试脚本:

#!/bin/bash

user_input=$1
while [ -z "$user_input" ]; do
        echo "please enter the name of the system you want to fix the game list for"
        echo "(as it is labelled in /home/pi/RetroPie/roms)"
        read -r user_input
done

ls "/home/pi/RetroPie/roms/$user_input" >/dev/null 2>&1

if  [ "$?" -ne 0 ]; then
        echo "this doesn't appear to be a system installed here. exiting."
        exit 1
fi

games_to_fix()
{
        IFS=$'\n'
        console=$1
        filepath="/opt/retropie/configs/all/emulationstation/gamelists/$console/gamelist.xml"
        game_array=($(fgrep "<name>" "$filepath" | sort | uniq -c | sort -rn | awk  '$1 > 1 {print $0}'| cut -d ">" -f 2 | cut -d "<" -f 1))
        number_to_fix=($(fgrep "<name>" "$filepath" | sort | uniq -c | sort -rn | awk  '$1 > 1 {print $1}'))
}

get_new_name()
{
        mYpath=$1
        new_name=$(echo $mYpath | cut -d ">" -f 2 | cut -d "<" -f 1 | sed -e 's/\.\///g' | sed -e 's/\.7z//g')
}

games_to_fix $user_input

IFS=$'\n'
index=0
for i in ${number_to_fix[@]}; do
        loop=1
        for game in ${game_array[@]:$index:$i}; do
        #       for ind in $(eval echo {1..$i}); do
                line_number=$(fgrep -n "<name>$game</name>"  $filepath | awk '{print $1}' | cut -d : -f 1 | sed -e "${loop}q;d")
                path_line_number=$(expr $line_number - 1 )
                path=$(sed "${path_line_number}q;d" $filepath | cut -d : -f 2)
                get_new_name "$path"
                sed -i "${line_number}s/$game/$new_name/g" $filepath
                ((loop++))
        done
        index=$(expr index + $i);
done

答案1

简单来说:您应该在数组扩展周围使用引号像这样,除非你明确想要字段/单词分割。"$@"将每个位置参数扩展为一个单独的单词,类似地也是如此"${a[@]}"。推而广之,对于"${a[@]:0:2}".


IFS也就是说,Bash 中似乎仍然存在不一致,并且您使用的内容应该适用于您的情况(因为值中没有全局字符,并且通过正确设置来处理字段分割)。

获取完整数组的效果:

$ IFS=$'\n'
$ a=("foo bar" "asdf ghjk")
$ printf "%s\n" ${a[@]}
foo bar
asdf ghjk

切片不适用于数组,但适用于$@

$ printf "%s\n" ${a[@]:0:2}
foo bar asdf ghjk

$ set -- "aa bb" "cc dd"
$ printf "%s\n" ${@:1:2}
aa bb
cc dd

它确实可以在 ksh 和 zsh 中工作,这强调了 Bash 在这里不一致(zsh 当然会有自己的等效语法)

$ ifs=$'\n' ksh -c 'IFS="$ifs"; a=("foo bar" "asdf ghjk"); printf "%s\n" ${a[@]:0:2}'
foo bar
asdf ghjk
$ ifs=$'\n' zsh -yc 'IFS="$ifs"; a=("foo bar" "asdf ghjk"); printf "%s\n" ${a[@]:0:2}'
foo bar
asdf ghjk

引用的版本也适用于 Bash,并且当您只想要原样的值时更好,因为您不需要依赖于IFS.IFS即使数组元素有空格,默认值在这里也能正常工作:

$ unset IFS                         # uses default of $' \t\n'
$ printf "%s\n" "${a[@]:0:2}"
foo bar
asdf ghjk

看起来好像未加引号的${a[@]:0:2}元素用空格连接起来,有点像在 Bash 中不发生分词的情况下发生的情况(例如str=${a[@]})。然后它像往常一样尝试用 分割结果IFS。例如,在这里,它在第二个数组元素中间的换行符上分割:

$ IFS=$'\n'
$ a=("foo bar" $'new\nline' "asdf ghjk");
$ printf ":%s\n" ${a[@]:0:3}
:foo bar new
:line asdf ghjk

如上所述,在大多数情况下,您确实应该在数组扩展周围使用引号,尽管人们仍然会假设${a[@]:n:m}会产生多个单词,就像${a[@]}这样做一样。

这里的行为似乎存在于 Bash4.4.12(1)-release5.0.0(1)-alpha.我发布了有关它的错误报告。

答案2

引用
引用

引用

引用

引用!!

这有效:

$ game_array=("foo bar" "baz" "bam foo" "bar")

$ for game in "${game_array[@]:0:10}" ; do echo "$game"; done
foo bar
baz
bam foo
bar

相关内容