命令替换

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

exp 1

1[root@localhost ~]# echo today is $(date "+%Y-%m-%d")
2today is 2017-11-07
3[root@localhost ~]# echo today is `date "+%Y-%m-%d"`
4today is 2017-11-07

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

``很容易与”搞混乱,尤其对初学者来说,而$( )比较直观。
最后,$( )的弊端是,并不是所有的类unix系统都支持这种方式,但反引号是肯定支持的。

exp 2

1[root@localhost ~]#  echo Linux `echo Shell `echo today is `date "+%Y-%m-%d"```
2Linux Shellecho today is 2017-11-07     #过多使用``会有问题
3[root@localhost ~]# echo Linux `echo Shell $(echo today is $(date "+%Y-%m-%d"))`
4Linux Shell today is 2017-11-07    ``$()混合使用
5[root@localhost ~]# echo Linux $(echo Shell $(echo today is $(date "+%Y-%m-%d")))
6Linux Shell today is 2017-11-07    #多个$()同时使用也不会有问题

${ }变量替换

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

exp 1

1[root@localhost ~]# A=Linux
2[root@localhost ~]# echo $AB    #表示变量AB
3
4[root@localhost ~]# echo ${A}B    #表示变量A后连接着B
5LinuxB

取路径、文件名、后缀

 1先赋值一个变量为一个路径,如下:
 2file=/dir1/dir2/dir3/my.file.txt
 3
 4命令    解释    结果
 5${file#*/}    拿掉第一条 / 及其左边的字符串    dir1/dir2/dir3/my.file.txt
 6[root@localhost ~]# echo ${file#*/}
 7dir1/dir2/dir3/my.file.txt
 8
 9${file##*/}    拿掉最后一条 / 及其左边的字符串    my.file.txt
10[root@localhost ~]# echo ${file##*/}
11my.file.txt
12
13${file#*.}    拿掉第一个 . 及其左边的字符串    file.txt
14[root@localhost ~]# echo ${file#*.}
15file.txt
16
17${file##*.}    拿掉最后一个 . 及其左边的字符串    txt
18[root@localhost ~]# echo ${file##*.}
19txt
20
21${file%/*}    拿掉最后一条 / 及其右边的字符串    /dir1/dir2/dir3
22[root@localhost ~]# echo ${file%/*}
23/dir1/dir2/dir3
24
25${file%%/*}    拿掉第一条 / 及其右边的字符串    (空值)
26[root@localhost ~]# echo ${file%%/*}
27(空值)
28
29${file%.*}    拿掉最后一个 . 及其右边的字符串    /dir1/dir2/dir3/my.file
30[root@localhost ~]# echo ${file%.*}
31/dir1/dir2/dir3/my.file
32
33${file%%.*}    拿掉第一个 . 及其右边的字符串    /dir1/dir2/dir3/my
34[root@localhost ~]# echo ${file%%.*}
35/dir1/dir2/dir3/my
36记忆方法如下:
37
38# 是去掉左边(在键盘上 # 在 $ 之左边)
39% 是去掉右边(在键盘上 % 在 $ 之右边)
40单一符号是最小匹配;两个符号是最大匹配
41*是用来匹配不要的字符,也就是想要去掉的那部分
42还有指定字符分隔号,与*配合,决定取哪部分

取子串及替换

1命令                                    解释                              结果
2${file:0:5}               提取最左边的 5 个字节                /dir1
3${file:5:5}               提取第 5 个字节右边的连续 5 个字节         /dir2
4${file/dir/path}            将第一个 dir 提换为 path              /path1/dir2/dir3/my.file.txt
5${file//dir/path}        将全部 dir 提换为 path               /path1/path2/path3/my.file.txt
6${#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 也受影响.

数组

1A="a b c def"   # 定义字符串
2A=(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]=xzy 则是将第四个组数重新定义为 xyz

$(( ))与整数运算

bash中整数运算符号

符号 功能
+ – * / 分别为加、减、乘、除
% 余数运算
& ^ ! | 分别为“AND、OR、XOR、NOT”

在 $(( )) 中的变量名称,可于其前面加 $ 符号来替换,也可以不用。

1[root@localhost ~]# echo $((2*3))
26
3[root@localhost ~]# a=5;b=7;c=2
4[root@localhost ~]# echo $((a+b*c))
519
6[root@localhost ~]# echo $(($a+$b*$c))
719

进制转换

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

1[root@localhost ~]# echo $((2#110))
26
3[root@localhost ~]# echo $((16#2a))
442
5[root@localhost ~]# echo $((8#11))
69

(())重定义变量值

 1[root@localhost ~]# a=5;b=7
 2[root@localhost ~]# ((a++))
 3[root@localhost ~]# echo $a
 46
 5[root@localhost ~]# ((a--));echo $a
 65
 7[root@localhost ~]# ((a<b));echo $?
 80
 9[root@localhost ~]# ((a>b));echo $?
101