${}基本功能
一般情況下$var與${var}是沒有區別的,但是用${ }會比較精確的界定變數名稱的范圍
[root@localhost ~]# A=Linux [root@localhost ~]# echo $AB #表示變數AB [root@localhost ~]# echo ${A}B #表示變數A后連接著B LinuxB
除此之外${}還有很多其他功能如下:
測驗變數宣告:
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 也受影響.
${}字串截取
示例如下:
命令 | 解釋 | 結果
${file#*.} 拿掉第一個 . 及其左邊的字串 file.txt
[root@localhost ~]# echo ${file#*.} file.txt
${file##*.} 拿掉最后一個 . 及其左邊的字串 txt
[root@localhost ~]# echo ${file##*.} txt
${file%.*} 拿掉最后一個 . 及其右邊的字串 /dir1/dir2/dir3/my.file
[root@localhost ~]# echo ${file%.*} /dir1/dir2/dir3/my.file
${file%%.*} 拿掉第一個 . 及其右邊的字串 /dir1/dir2/dir3/my
[root@localhost ~]# echo ${file%%.*} /dir1/dir2/dir3/my
記憶方法如下:
- # 是去掉左邊(在鍵盤上 # 在 $ 之左邊)
- % 是去掉右邊(在鍵盤上 % 在 $ 之右邊)
- 單一符號是最小匹配;兩個符號是最大匹配
- *是用來匹配不要的字符,也就是想要去掉的那部分
- 還有指定字符分隔號,與*配合,決定取哪部分
上面是網上的解釋,按照上面示例,好像${}只能用單字符定界,實際可以用字串,并且也不是只能匹配洗掉定界字串的一側,實際可以兩邊同時匹配洗掉(但是因為存在行首或行尾定界,#模式指定匹配行首必洗掉行首一側,%模式同理,看下面解釋)
補充:經測驗發現,實際上這里#和%指代的是匹配字串開頭或者結尾,指定匹配正則中的^或者$,連續的#和$,類似于正則的貪婪匹配和最短匹配,*相當于標準正則中的.* ,類似于的shell命令中的匹配模式, ${}實際上就是以這種非標準正則的方式匹配字串,然后截取洗掉,
示例如下:
${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 |
- 實際上在shell中變數A和陣列元素A[0]是等價的
單獨宣告變數后,同樣可以以陣列方式取值
[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
參考補充自(https://www.cnblogs.com/chengd/p/7803664.html)
轉載請註明出處,本文鏈接:https://www.uj5u.com/caozuo/68284.html
標籤:Linux
下一篇:linux檔案讀寫權限
