通常状况下$var与${var}是没有区别的,可是用${ }会比较精确的界定变量名称的范围html
[root@localhost ~]# A=Linux [root@localhost ~]# echo $AB #表示变量AB [root@localhost ~]# echo ${A}B #表示变量A后链接着B LinuxB
除此以外${}还有不少其余功能以下:
测试变量声明:shell
file=/dir1/dir2/dir3/my.file.txt
命令 | 解释 | 结果 |
---|---|---|
${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 也受影响.bash
示例以下:
命令 | 解释 | 结果
${file#*.} 拿掉第一个 . 及其左边的字符串 file.txt测试
[root@localhost ~]# echo ${file#*.} file.txt
${file##*.} 拿掉最后一个 . 及其左边的字符串 txtspa
[root@localhost ~]# echo ${file##*.} txt
${file%.*} 拿掉最后一个 . 及其右边的字符串 /dir1/dir2/dir3/my.filecode
[root@localhost ~]# echo ${file%.*} /dir1/dir2/dir3/my.file
${file%%.*} 拿掉第一个 . 及其右边的字符串 /dir1/dir2/dir3/myhtm
[root@localhost ~]# echo ${file%%.*} /dir1/dir2/dir3/my
上面是网上的解释,按照上面示例,好像${}只能用单字符定界,实际能够用字符串,而且也不是只能匹配删除定界字符串的一侧,实际能够两边同时匹配删除(可是由于存在行首或行尾定界,#模式指定匹配行首必删除行首一侧,%模式同理,看下面解释)blog
补充:经测试发现,实际上这里#和%指代的是匹配字符串开头或者结尾,指定匹配正则中的^或者$。连续的#和$,相似于正则的贪婪匹配和最短匹配,*至关于标准正则中的.* ,相似于的shell命令中的匹配模式。 ${}实际上就是以这种非标准正则的方式匹配字符串,而后截取删除。ip
示例以下:
${file##/*.fi} 最长匹配,可是没有匹配到后面的"." ,由于这里匹配的是"/*.fi",.fi与.txt不匹配,而前面的/*
等价于正则的^/*,能够匹配字符串前面部分。
[root@test ~]# echo ${file##/*.fi} le.txt
若是换成{file##d*.},由于是匹配字符串开头模式,前面的d*等价于匹配正则的^d*,匹配失败返回原字符串
[root@test ~]# echo ${file##d*.} /dir1/dir2/dir3/my.file.txt
${file#*.*t} 最短匹配,这里匹配是"*.*t",能够匹配.txt前面部分,结果以下
[root@test ~]# echo ${file#*.*t} xt
${file%dir*xt} 匹配行尾部分
[root@test ~]# echo ${file%dir*xt} /dir1/dir2/
这里的*也不是必要的,没有一样能够匹配成功
[root@test ~]# echo ${file%le.txt} /dir1/dir2/dir3/my.fi
所以${}中#和$就是用相似正则的方式指定匹配行首或者行尾匹配删除字符
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]=xzy | 则是将第四个组数从新定义为 xyz |
单独声明变量后,一样能够以数组方式取值
[root@test ~]# B="12 3 4" [root@test ~]# echo ${B[*]} 12 3 4 [root@test ~]# echo ${#B[*]} 1 [root@test ~]# echo ${B[0]} 12 3 4