私はRetropieで重複する命名問題を解決する簡単なbashスクリプトを書いています。
スクリプト自体は、gameslist.xmlファイルから複数回参照された名前を取得し、後で使用するために配列に保存します。
私は次のようにインデックスを介してこの配列を繰り返しました。
pi@retropie:~ $ for game in ${game_array[@]:0:10} ; do echo $game; done
最初から10番目の要素(たとえば)を取得しますが、${game_array[9]}
出力は1行にリンクされます。
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
フィールド区切り文字がnew lineに設定されているIFS='$\n'
ため、2番目は機能しますが、最初は機能しない理由を一生にわかりませんか?
コンテキストの完全なテストスクリプトは次のとおりです。
#!/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
。たとえば、ここでは2番目の配列要素の中央にある改行文字に分割されます。
$ 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[@]}
ここの動作はBashに存在しているように見え、4.4.12(1)-release
それ5.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