Zsh 開發指南(第十一篇 變數的進階內容)

陌辭寒發表於2018-10-26

導讀

之前我們已經依次講過 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 中已經沒有那些問題了,變數兩邊無需加雙引號,不會出現莫名其妙的錯誤。但有些地方還是需要加雙引號的。

需要加雙引號的場景:

  1. 像這樣的包含字元或者特殊符號的字串 "aa bb
    *"
    出現在程式碼中時,兩邊要加雙引號,這個基本不需要說明。
  2. 在用 $() 呼叫命令時,如果希望結果按一個字串處理,需要加上雙引號,"$()",不然的話,如果命令結果中有空格,$() 會被展開成多個字串。
  3. 如果想將陣列當單個字串處理,需要加雙引號,array=(a b); print -l "$array"
  4. 其他的原本不是單個字串的東西,需要轉成單個字串的場景,要加雙引號。

其餘情況通常都不需要加雙引號,典型的情況:

  1. 任何情況下,字串變數的兩邊都不需要加雙引號,無論裡邊的內容多麼特殊,或者變數存不存在,都沒有關係,如 $str
  2. 如果不轉換型別(比如陣列轉成字串),任何變數的兩邊都不需要加雙引號。
  3. $1 $2 $* 這些引數(其實它們也都是單個字串),都不需要加雙引號,無論內容是什麼,或者引數是否存在。

以上的 7 種情況幾乎覆蓋了所有場景,如果有沒覆蓋到的,試一下即可(讓裡邊的內容包含空格、換行和其他特殊字元等等,看看結果是否符合預期)。

總結

本文簡單介紹了一些比較使用的 typeset(或者 local)命令的用法,typeset 命令還有很多其他引數,但一般很少用,以後我也會繼續更新。

參考

www.bash2zsh.com/zsh_refcard…

www.linux-mag.com/id/1079/

全系列文章地址:github.com/goreliu/zsh…

付費解決 Windows、Linux、Shell、C、C++、AHK、Python、JavaScript、Lua 等領域相關問題,靈活定價,歡迎諮詢,微信 ly50247。

相關文章