在bash中,​​$( )​​​与` `(反引号)都是用来作命令替换的。
命令替换与变量替换差不多,都是用来重组命令行的,先完成引号里的命令行,然后将其结果替换出来,再重组成新的命令行。

例子:

$ echo today is $(date "+%Y-%m-%d")
today is 2014-07-01

$( )与 ` `

在操作上,这两者都是达到相应的效果,但是建议使用​​$()​​,理由如下:

  • ““很容易与”搞混乱,尤其对初学者来说。
  • 在多层次的复合替换中,​​```必须要额外的跳脱处理(反斜线),而​​$()`比较直观。
  • 最后,​​$()​​的弊端是,并不是所有的类unix系统都支持这种方式,但反引号是肯定支持的。

例子:

# 将cmd1执行结果作为cmd2参数,再将cmd2结果作为cmd3的参数
cmd3 $(cmd2 $(cmd1))

# 如果是用反引号,直接引用是不行的,还需要作跳脱处理

​${ }​​变量替换

一般情况下,​​$var​​​与​​${var}​​​是没有区别的,但是用​​${ }​​会比较精确的界定变量名称的范围

例子:

$ A=B
$ echo ${A}B
BB

取路径、文件名、后缀

先赋值一个变量为一个路径,如下:
​​​file=/dir1/dir2/dir3/my.file.txt​

命令

解释

结果

​${file#*/}​

拿掉第一条 / 及其左边的字符串

dir1/dir2/dir3/my.file.txt

​${file##*/}​

拿掉最后一条 / 及其左边的字符串

my.file.txt

​${file#*.}​

拿掉第一个 . 及其左边的字符串

file.txt

​${file##*.}​

拿掉最后一个 . 及其左边的字符串

txt

​${file%/*}​

拿掉最后一条 / 及其右边的字符串

/dir1/dir2/dir3

​${file%%/*}​

拿掉第一条 / 及其右边的字符串

(空值)

​${file%.*}​

拿掉最后一个 . 及其右边的字符串

/dir1/dir2/dir3/my.file

​${file%%.*}​

拿掉第一个 . 及其右边的字符串

/dir1/dir2/dir3/my

记忆方法如下:

  • ​#​​ 是去掉左边(在键盘上 # 在 $ 之左边)
  • ​%​​ 是去掉右边(在键盘上 % 在 $ 之右边)
  • 单一符号是最小匹配;两个符号是最大匹配
  • ​*​​是用来匹配不要的字符,也就是想要去掉的那部分
  • 还有指定字符分隔号,与*配合,决定取哪部分

取子串及替换

命令

解释

结果

​${file:0:5}​

提取最左边的 5 个字节

/dir1

​${file:5:5}​

提取第 5 个字节右边的连续 5 个字节

/dir2

​${file/dir/path}​

将第一个 dir 提换为 path

/path1/dir2/dir3/my.file.txt

​${file//dir/path}​

将全部 dir 提换为 path

/path1/path2/path3/my.file.txt

​${#file}​

获取变量长度

27

根据状态为变量赋值

命令

解释

结果

​${file-my.file.txt}​

若 $file 没设定,则使用 my.file.txt 作传回值

空值及非空值不作处理

​${file:-my.file.txt}​

若 $file 没有设定或为空值,则使用 my.file.txt

作传回值 非空值时不作处理

​${file+my.file.txt}​

若$file 设为空值或非空值,均使用my.file.txt作传回值

没设定时不作处理

​${file:+my.file.txt}​

若 $file 为非空值,则使用 my.file.txt 作传回值

没设定及空值不作处理

​${file=txt}​

若 file没设定,则回传txt,并将file 赋值为 txt

空值及非空值不作处理

​${file:=txt}​

若 file没设定或空值,则回传txt,将file

赋值为txt 非空值时不作处理

​${file?my.file.txt}​

若 $file 没设定,则将 my.file.txt

输出至 STDERR 空值及非空值不作处理

​${file:?my.file.txt}​

若 $file没设定或空值,则将my.file.txt输出至STDERR

非空值时不作处理

tips:
以上的理解在于, 你一定要分清楚 unset 与 null 及 non-null 这三种赋值状态. 一般而言, : 与 null 有关, 若不带 : 的话, null 不受影响, 若带 : 则连 null 也受影响.

数组

A="a b c def"   # 定义字符串
A=(a b c def) # 定义字符数组

命令

解释

结果

​${A[@]}​

返回数组全部元素

a b c def

​${A[*]}​

同上

a b c def

​${A[0]}​

返回数组第一个元素

a

​${#A[@]}​

返回数组元素总个数

4

​${#A[*]}​

同上

4

​${#A[3]}​

返回第四个元素的长度,即def的长度

3

​A[3]=xyz​

则是将第四个组数重新定义为

xyz

$(( ))与整数运算

bash中整数运算符号

符号

功能

​+ - * /​

分别为加、减、乘、除

​%​

余数运算

​& ^ !​

分别为“AND、OR、XOR、NOT”

在 (())中的变量名称,可于其前面加

$ a=5;b=7;c=2
$ echo $((a+b*c))
19
$ echo $(($a+$b*$c))
19

进制转换

​$(( ))​​​可以将其他进制转成十进制数显示出来。用法如下:
​​​echo $((N#xx))​​​
其中,N为进制,xx为该进制下某个数值,命令执行后可以得到该进制数转成十进制后的值。

$ echo $((2#110))   # 二进制转十进制
6
$ echo $((16#2a)) # 十六进制转十进制
42
$ echo $((8#11)) # 八进制转十进制
9

(( ))重定义变量值

$ a=5;b=7
$ ((a++));echo $a
6
$ ((a--));echo $a
5
$ ((a<b));echo $?
0

使用(( ))作整数测试时,不要跟[ ]的整数测试搞混乱了。

参考

​http://blog.sina.com.cn/s/blog_4da051a60102uwda.html​