举个栗子:
v=jfedu.shiyiwen.com echo ${v%.*}
一个%为非贪婪匹配,即匹配最短结果。%从右到左进行非贪婪匹配,匹配什么呢? 匹配已 .*的字符。 那么当然是匹配到了.com 然后执行删除匹配到的字符。结果就是(这里的 "." 相当于定界符。而“*”是通配符)
jfedu.shiyiwen
echo ${v%%.*}
而这里执行贪婪匹配,即匹配最长结果就匹配到了 .shiyiwen.com 结果为
jfedu
这里在说一个 # 刚好和 %顺序相反而已,#是从左到右来进行匹配的。 老规矩:举个栗子
v=jfedu.shiyiwen.com echo ${v#*.} shiyiwen.com echo ${v##*.} com
用处有很多,比较常见的就是很方便的获取文件后缀名和文件名。举个栗子
01.#!/bin/bash 02. 03.num=1 04.for i in *.tar *.tar.gz 05.do 06.new=new_$num.${i#*.} 07.mv $i $new 2>/dev/null 08. 09.if [ $? -eq 0 ];then 10. echo "remove $i to $new" 11. let num++ 12.fi 13. 14.done
在补充一些:
${#VALUE}:计算VALUE字符串的字符数量。
- ${VALUE%.*}或${VALUE%%.*}:删除VALUE字符串中以分隔符“.”匹配的右边字符,保留左边字符。
- ${VALUE#*.}或${VALUE##*.}:删除VALUE字符串中以分隔符“.”匹配的左边字符,保留右边字符。
- ${VALUE/OLD/NEW}或${VALUE//OLD/NEW}:用NEW子串替换VALUE字符串中匹配的OLD子串。
补充:“*”表示通配符,用于匹配字符串将被删除的字串。“.”表示字符串中分隔符,可以为任意一个或多个字符。“%”表示从右向左匹配,“#”表示从左向右匹配,“”表示替换,都属于非贪婪匹配,即匹配符合通配符的最短结果。与“%”、“#”和“/”类似的有“%%”、“##”和“//”,都属于贪婪匹配,即匹配符合通配符的最长结果。
- ${VALUE:OFFSET}或${VALUE:OFFSET:LENGTH}:从VALUE字符串的左边开始中截取子串。
- ${VALUE:0-OFFSET}或${VALUE:0-OFFSET:LENGTH}:从VALUE字符串的右边开始中截取子串。
补充:左边第一个字符从“0”开始,右边第一个字符从“0-1”开始。 表示偏移OFFSET个字符开始,LENGTH表示要截取字符的长度。如果没有LENGTH变量,表示偏移OFFSET个字符开始到字符串结束。
- ${VALUE:-WORD}:当变量未定义或者值为空时,返回值为WORD的内容,否则返回变量的值。
- ${VALUE:=WORD}:当变量未定义或者值为空时,返回WORD的值的同时并将WORD赋值给VALUE,否则返回变量的值。
- ${VALUE:+WORD}:当变量已赋值时,其值才用WORD替换,否则不进行任何替换。
- ${VALUE:?MESSAGE}:当变量已赋值时,正常替换。否则将消息MESSAGE送到标准错误输出(若此替换出现在SHELL程序中,那么该程序将终止运行)。
补充:WORD可以为一个字符串,也可以为一个变量。当为变量时,需要用“$”引用该变量。
另外 ${}的用途,请查看博客的另外一篇博文。