在shell脚本中,当展开变量时,我们什么时候使用{}?

例如,我已经看到以下:

var=10        # Declare variable

echo "${var}" # One use of the variable
echo "$var"   # Another use of the variable

有显著的区别吗,还是只是风格不同?两者孰优孰劣?


当前回答

在这个特殊的例子中,没有区别。然而,如果你想在字符串中扩展变量foo, ${}中的{}是有用的

"${foo}bar"

因为“$foobar”将展开由foobar标识的变量。

大括号在以下情况下也是无条件需要的:

展开数组元素,如${array[42]} 使用参数展开操作,如${filename%。*}(删除扩展名) 扩展超过9的位置参数:"$8 $9 ${10}${11}"

在任何地方都这样做,而不是只在可能有歧义的情况下这样做,可以被认为是很好的编程实践。这既是为了一致性,也是为了避免像$foo_$bar.jpg这样的意外,在这种情况下,下划线成为变量名的一部分在视觉上并不明显。

其他回答

变量名的末尾通常用空格或换行符表示。但是如果我们在打印变量值后不需要空格或换行符怎么办?花括号告诉shell解释器变量名的结尾在哪里。

经典示例1)-没有尾随空格的shell变量

TIME=10

# WRONG: no such variable called 'TIMEsecs'
echo "Time taken = $TIMEsecs"

# What we want is $TIME followed by "secs" with no whitespace between the two.
echo "Time taken = ${TIME}secs"

例2)带有版本控制的jar的Java类路径

# WRONG - no such variable LATESTVERSION_src
CLASSPATH=hibernate-$LATESTVERSION_src.zip:hibernate_$LATEST_VERSION.jar

# RIGHT
CLASSPATH=hibernate-${LATESTVERSION}_src.zip:hibernate_$LATEST_VERSION.jar

(Fred的回答已经说明了这一点,但他的例子有点太抽象了)

你也可以在括号内做一些文本操作:

STRING="./folder/subfolder/file.txt"
echo ${STRING} ${STRING%/*/*}

结果:

./folder/subfolder/file.txt ./folder

or

STRING="This is a string"
echo ${STRING// /_}

结果:

This_is_a_string

你说得对,“常规变量”是不需要的……但是它对调试和阅读脚本更有帮助。

根据SierraX和Peter关于文本操作的建议,花括号{}用于将变量传递给命令,例如:

假设你有一个position .txt文件,其中包含一本著名的意大利小说的第一行:

> sposi="somewhere/myfolder/sposi.txt"
> cat $sposi

科莫湖中午的那根树枝

现在创建两个变量:

# Search the 2nd word found in the file that "sposi" variable points to
> word=$(cat $sposi | cut -d " " -f 2)

# This variable will replace the word
> new_word="filone"

现在将word变量content替换为position .txt文件中的new_word变量content

> sed -i "s/${word}/${new_word}/g" $sposi
> cat $sposi

科莫湖中午的水流

“ramo”一词已被取代。

花括号总是用于访问数组元素和执行大括号展开。

不要过于谨慎,使用{}进行shell变量扩展,即使在不存在歧义的情况下也是如此。

例如:

dir=log
prog=foo
path=/var/${dir}/${prog}      # excessive use of {}, not needed since / can't be a part of a shell variable name
logfile=${path}/${prog}.log   # same as above, . can't be a part of a shell variable name
path_copy=${path}             # {} is totally unnecessary
archive=${logfile}_arch       # {} is needed since _ can be a part of shell variable name

所以,这三行最好写成:

path=/var/$dir/$prog
logfile=$path/$prog.log
path_copy=$path

这绝对更有可读性。

由于变量名不能以数字开头,shell不需要在有编号的变量(如$1,$2等)周围加上{},除非变量展开后跟着一个数字。这太微妙了,它确实使得在这样的上下文中显式地使用{}:

set app      # set $1 to app
fruit=$1le   # sets fruit to apple, but confusing
fruit=${1}le # sets fruit to apple, makes the intention clear

See:

Linux环境变量名中允许使用的字符

使用{}进行分组。对数组元素进行解引用时需要使用大括号。例子:

dir=(*)           # store the contents of the directory into an array
echo "${dir[0]}"  # get the first entry.
echo "$dir[0]"    # incorrect