次のプログラムでは、最初のステートメント内で変数$foo
を値 1 に設定するとif
、if ステートメントの後にその値が記憶されるという意味で動作します。ただし、ステートメント内の an 内で同じ変数を値 2 に設定するとif
、ループwhile
後に忘れられます。ループ内でwhile
変数の何らかのコピーを使用して、その特定のコピーのみを変更しているかのように動作します。完全なテスト プログラムは次のとおりです。$foo
while
#!/bin/bash
set -e
set -u
foo=0
bar="hello"
if [[ "$bar" == "hello" ]]
then
foo=1
echo "Setting \$foo to 1: $foo"
fi
echo "Variable \$foo after if statement: $foo"
lines="first line\nsecond line\nthird line"
echo -e $lines | while read line
do
if [[ "$line" == "second line" ]]
then
foo=2
echo "Variable \$foo updated to $foo inside if inside while loop"
fi
echo "Value of \$foo in while loop body: $foo"
done
echo "Variable \$foo after while loop: $foo"
# Output:
# $ ./testbash.sh
# Setting $foo to 1: 1
# Variable $foo after if statement: 1
# Value of $foo in while loop body: 1
# Variable $foo updated to 2 inside if inside while loop
# Value of $foo in while loop body: 2
# Value of $foo in while loop body: 2
# Variable $foo after while loop: 1
# bash --version
# GNU bash, version 4.1.10(4)-release (i686-pc-cygwin)
ベストアンサー1
echo -e $lines | while read line
...
done
ループwhile
はサブシェルで実行されます。そのため、変数に加えた変更は、サブシェルが終了すると使用できなくなります。
代わりに、ここで文字列while ループをメインシェルプロセス内に書き直すと、echo -e $lines
サブシェル内でのみ実行されます。
while read line
do
if [[ "$line" == "second line" ]]
then
foo=2
echo "Variable \$foo updated to $foo inside if inside while loop"
fi
echo "Value of \$foo in while loop body: $foo"
done <<< "$(echo -e "$lines")"
echo
上記の here-string のかなり醜い部分は、 を割り当てるときにすぐにバックスラッシュ シーケンスを展開することで取り除くことができますlines
。$'...'
引用符の形式はここで使用できます。
lines=$'first line\nsecond line\nthird line'
while read line; do
...
done <<< "$lines"