導讀
之前我們已經依次講過 zsh 下的五種變量(字符串、數組、哈希表、整數、浮點數)的基本用法。但變量的使用方面,還有一些比較進階的內容,這對一些比較特別的場景很有幫助。
typeset 命令
typeset 命令用于對變量進行詳細的設置。我們之前在哈希表那篇見過它。typeset -A 可以用來定義哈希表。
% typeset -A hashmap=(aa bb cc dd)
但我們后續都使用 local,因為 local 的功能和 hashmap 是一樣的(除了不能用 -f 和 -g,這兩個選項不常用),并且更短更容易輸入。這里提到 typeset 命令,因為這個名稱很好地反映了它的功能。但知道了這個后,我們可以繼續使用 local 命令,畢竟它們是一樣的。
typeset 命令有很多選項,可以作用在變量上,起到各種各樣的效果。
強制字符串內容為小寫或者大寫
# 強制字符串內容為小寫
% local -l str=abcABC && echo $str
abcabc
# 強制字符串內容為大寫
% local -u str=abcABC && echo $str
ABCABC
設置變量為環境變量
% local -x str=abc
# 通常使用 export,功能一樣
% export str=abc
環境變量可以被子進程讀取。
設置變量為只讀變量
% local -r str1=abc
# 通常使用 readonly,功能一樣
% readonly str2=abc
% str1=bcd
zsh: read-only variable: str1
% str2=bcd
zsh: read-only variable: str2
設置數組不包含重復元素
% local -U array=(aa bb aa cc) && echo $array
aa bb cc
設置整數的位數
# 如果位數不夠,輸出內容會用 0 補全
% local -Z 3 i=5 && echo $i
005
# 如果超出范圍會被截斷
% local -Z 3 i=1234 && echo $i
234
進制轉換
設置整數為其他進制顯示:
% local -i 16 i=255
% echo $i
16#FF
可以設置 2 到 36 之間任意進制。設置幾進制顯示,并不影響計算,只是顯示格式不同。
用 [#n] num 也可以顯示十進制數為 n 進制:
% echo $(([#16] 255))
16#FF
可以用 n#num 來顯示 n 進制整數為十進制:
% echo $((16#ff))
255
我們可以定義一系列函數來快捷地轉換進制,不需要使用 bc 等外部命令:
0x() {
echo $((16#$1))
}
0o() {
echo $((8#$1))
}
0b() {
echo $((2#$1))
}
p16() {
echo $(([#16] $1))
}
p8() {
echo $(([#8] $1))
}
p2() {
echo $(([#2] $1))
}
# 其他進制轉十進制
% 0x ff
255
% 0b 1101
13
# 十進制轉其他進制
% p16 1234
16#4D2
同時對多個變量賦相同的值
% local {i,j,k}=123
% echo $i $j $k
123 123 123
綁定字符串和數組
% local -T DIR dir
% dir=(/a /b/c /b/d /e/f)
% echo $DIR
/a:/b/c:/b/d:/e/f
# 刪除 dir 后,DIR 也會被刪除(反之亦然)
% unset dir
% echo $+DIR
0
Linux 下經常需要處理帶分隔符冒號的字符串(比如 $PATH)。如果只修改其中某一個字段,比較麻煩。local -T 可以把字符串綁定到數組上,這樣直接修改數組,字符串內容也會同步變化(反之亦然)。其實在 zsh 中,$PATH 字符串就是和 $path 數組綁定的,可以直接通過修改 $path 來達到修改 $PATH 的目的,這在某些場景會方便很多。
顯示變量的定義方式
% array=(aa bb cc)
% local -p array
typeset -a array=(aa bb cc)
% array+=(dd)
% local -p array
typeset -a array=(aa bb cc dd)
什么地方該加雙引號
用過 bash 的讀者大概會對里邊的雙引號印象比較深刻,很多地方不加雙引號都會出錯,為了避免出錯,很多人每個變量左右都加上雙引號,麻煩不說,代碼看起來也比較亂。
其實 zsh 中已經沒有那些問題了,變量兩邊無需加雙引號,不會出現莫名其妙的錯誤。但有些地方還是需要加雙引號的。
需要加雙引號的場景:
- 像這樣的包含字符或者特殊符號的字符串
"aa bb \t \n *"
出現在代碼中時,兩邊要加雙引號,這個基本不需要說明。 - 在用
$()
調用命令時,如果希望結果按一個字符串處理,需要加上雙引號,"$()"
,不然的話,如果命令結果中有空格,$()
會被展開成多個字符串。 - 如果想將數組當單個字符串處理,需要加雙引號,
array=(a b); print -l "$array"
。 - 其他的原本不是單個字符串的東西,需要轉成單個字符串的場景,要加雙引號。
其余情況通常都不需要加雙引號,典型的情況:
- 任何情況下,字符串變量的兩邊都不需要加雙引號,無論里邊的內容多么特殊,或者變量存不存在,都沒有關系,如
$str
。 - 如果不轉換類型(比如數組轉成字符串),任何變量的兩邊都不需要加雙引號。
-
$1
$2
$*
這些參數(其實它們也都是單個字符串),都不需要加雙引號,無論內容是什么,或者參數是否存在。
以上的 7 種情況幾乎覆蓋了所有場景,如果有沒覆蓋到的,試一下即可(讓里邊的內容包含空格、換行和其他特殊字符等等,看看結果是否符合預期)。
總結
本文簡單介紹了一些比較使用的 typeset(或者 local)命令的用法,typeset 命令還有很多其他參數,但一般很少用,以后我也會繼續更新。
參考
http://www.bash2zsh.com/zsh_refcard/refcard.pdf
http://www.linux-mag.com/id/1079/
更新歷史
20170831:新增“什么地方該加雙引號”
本文不再更新,全系列文章在此更新維護:github.com/goreliu/zshguide
付費解決 Windows、Linux、Shell、C、C++、AHK、Python、JavaScript、Lua 等領域相關問題,靈活定價,歡迎咨詢,微信 ly50247。