shell變量操作${}詳細用法


 

${}基本功能

一般情況下$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) 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM