${}

${变量#关键词} 若变量内容从头开始的数据符合『关键词』,则将符合的最短数据初除 
${变量##关键词} 若变量内容从头开始的数据符合『关键词』,则将符合的最长数据初除 
${变量%关键词} 若变量内容从尾向前的数据符合『关键词』,则将符合的最短数据初除 
${变量%%关键词} 若变量内容从尾向前的数据符合『关键词』,则将符合的最长数据初除 
${变量/旧字符串/新字符串} 若变量内容符合『旧字符串』则『第一个旧字符串会被新字符串叏代』
${变量//旧字符串/新字符串} 若变量内容符合『旧字符串』则『全部的旧字符串会被新字符串叏代』

例:$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
记忆的方法为:
# 是去掉左边(在键盘上 # 在 $ 之左边)
% 是去掉右边(在键盘上 % 在 $ 之右边)
单一符号是最小匹配﹔两个符号是最大匹配。

其实,在“#”后面,无非就是一个匹配问题,不限于两个,你可以放任意个字符,还可以用shell中另外的通配符“?”“[…]”“[!…]”,例如:
$ echo ${file#????}
1/dir2/dir3/my.file.txt
$ echo ${file#*[0-9]}
/dir2/dir3/my.file.txt
$ echo ${file#/dir1/dir[0-9]}
/dir3/my.file.txt
“#”:相当于最小匹配,遇到一个最小的符合其后表达式的字符串(单个或多个)即中止匹配动作;
“##”:相当于最大匹配,它尽可能的匹配更多的字符。
我们可以拿“*”来说明:  
*  在shell中表示匹配任何符号包括空。当它在只有一个 # 的变量替换中,受最小匹配的影响,它不会匹配任何可打印字符,只匹配一个空,也就是什么也不匹配,你完全可以忽略它的存在;
当在有两个 ## 的变量替换中,受最大匹配的影响,一个 * 表示匹配整个字符串
如果想匹配字符“*”时,要在“*”前加一个“”,其后的“*”失去通配符的功能。

${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
====================================================
 
利用 ${ } 还可针对不同的变量状态赋值(没设定、空值、非空值即unset 和 null 及 non-null): 
------------------------------------------------------------------------------------
变量设定方式 str没有设定 str为空字符串 str已设定为非空字符串 
var=${str-expr}         var=expr   var=   var=$str 
var=${str:-expr}       var=expr   var=expr   var=$str 
var=${str+expr}       var=   var=expr   var=expr 
var=${str:+expr}       var= var=   var=expr 
----------------------------------------------------------------------------------
var=${str=expr} str=expr str 不变 str 不变 
var=expr var= var=$str 
----------------------------------------------------------------------------------
var=${str:=expr} str=expr str=expr str 不变 
var=expr var=expr var=$str 
----------------------------------------------------------------------------------
var=${str?expr}        expr 输出至 stderr    var= var=$str 
var=${str:?expr}      expr 输出至 stderr     expr 输出至 stderr     var=$str 
-----------------------------------------------------------------------------------

用${}来进行扩展变量替换的时候无外乎进行三种操做:
  1.返回替换的值
  2.更改变量本身的值
  3.输出信息到stderr

用各个不同的符号的时候,其结果是这三种操作的组合.
而只有在使用包涵=号的符号 (即${=}与 ${:=}) 时才会更改变量本身的值,其余的影响返回值。
以上的理解在于, 你一定要分清楚 unset 与 null 及 non-null 这三中赋值值状态.
一般而言, : 與 null 有关, 若不帶 : 的话, null 不受影响, 若带 : 則连 null 也受影响;也就是说,带 : 的话,unset 和 null 值都会受影响
===================================================
 

${#var} 可计算出变量值的长度:
${#file} 可得到 27 ,因为 /dir1/dir2/dir3/my.file.txt 刚好是 27 个字节...

 接下来,再为大家介稍一下 bash 的组数(array)处理方法。
一般而言,A="a b c def" 这样的变量只是将 $A 替换为一个单一的字符串,
但是改为 A=(a b c def) ,则是将 $A 定义为组数...

 bash 的组数替换方法可参考如下方法:
${A[@]} 或 ${A} 可得到 a b c def (全部组数) 
${A[0]} 可得到 a (第一个组数),${A[1]} 则为第二个组数... 
${#A[@]} 或 ${#A} 可得到 4 (全部组数数量) 
${#A[0]} 可得到 1 (即第一个组数(a)的长度),${#A[3]} 可得到 3 (第四个组数(def)的长度) A[3]=xyz 则是将第四个组数重新定义为 xyz ... 
原文地址:https://www.cnblogs.com/cool4ever/p/5868275.html