有段时间没有写shell了,今天偶尔看到一个shell文件内容。里面有些特殊的用法,只觉得有印象但就是说不出具体的功能,就又查了一下。顺便记录下来,防备以后参考。

首先,从脚本之家的一篇文章中借鉴些东西(http://www.jb51.net/article/60326.htm),说说shell中的各种括号,可以列举的有以下几种:

  1. ${var}

  2. $(cmd)

  3. ${var:-string},${var:+string},${var:=string},${var:?string}

  4. ${var%pattern},${var%%pattern},${var#pattern},${var##pattern}

我故意跳掉了两个,个人认为那个很平常,并且用处不多。

${var}只是$var的变形,不过前者比后者更明显的标示了变量名。从我看到的脚本中,很多人会使用${var}的形式。


$(cmd)用来执行cmd,返回结果。还有一个中更老的版本是`cmd`(使用的是反引号,ESC下面的那个),记得在哪里看到过说是从字面上看跟字符串的区分度太低,所以逐渐废弃了。转而使用$(cmd)形式。不过,从我看到的脚本`cmd`这种形式更多(大概因为是之前留存的吧)。


${var:-string},${var:+string},${var:=string},${var:?string}这个系列的,可以称之为条件赋值。只是条件和行为不一样。

${var:-string}当var为空时,返回string,但var变量本身仍然为空

${var:=string}当var为空时,返回string,但var变量也被赋值为string(感觉有点类似函数的默认参数)

${var:+string}当var不为空时,返回string,但var变量本身不变。当var为空时,表达式返回空

${var:?string}当var为空时,将string返回至stderr,并结束shell


${var%pattern},${var%%pattern},${var#pattern},${var##pattern}这个系列的,用来按照pattern去除变量的特定部分。

${var%pattern},${var%%patten}是从右开始去除,${var#pattern},${var##patten}是从左开始去除。${var%pattern},${var#patten}表示按照最短进行去除,${var%%pattern},{var##patten}标示按照最长匹配进行去除。如果不是用通配符,就没有最短和最长的概念。

常见的是用来从一个全路径中获取文件名和文件所在的路径。

path=$HOME/public/foo.sh

${path##*/}将返回文件名foo.sh,去除了左边的按照*/能找到的最长的匹配项

${path%/*}将返回文件所在目录$HOME/public(实际返回的值中$HOME会被替换为真实路径),去除了右边的最短匹配项


另一部分就是$打头的各种预定义变量。

$$脚本运行的PID,一般用于生成log

$?脚本执行结果

$#表示脚本参数的个数

"$*"将所有的脚本参数放入一个字符串中,使用IFS分隔各个参数。例如“1 2 3”

"$@"将所有的脚本参数放入一个字符串组中。例如"1""2""3"

$n标示第n个脚本参数(0<=n<=9),大于9个的参数使用${n}来获取


最后记一下变量嵌套,这个是后需要将内层的$变为!。

引用《linux command line and shell scripting bible》中的例子。

想要输出脚本的最后一个参数。最后一个参数序号可以通过$#获取,任何一个参数可以通过${n}来获取。但在{}中是不能够是用$的,需要使用!替换。所以${$#}要变成${!#}才能得到正确的输出。嵌套的变量使用一般都要在内层使用!var的形式


先写到这里,后面继续补充吧。