這次讓我們暫時丟開 command line ,先來了解一下 bash 變量(variable)吧...

所謂的變量,就是利用一個特定的"名稱"(name)來存取一段可以變化的"值"(value)。

*設定(set)*
在 bash 中,你可以用 "=" 來設定或重新定義變量的內容:
        name=value
在設定變量的時侯,得遵守如下規則:
        * 等號左右兩邊不能使用區隔符號(IFS),也應避免使用 shell 的保留字元(meta charactor)。
        * 變量名稱不能使用 $ 符號。
        * 變量名稱的第一個字母不能是數字(number)。
        * 變量名稱長度不可超過 256 個字母。
        * 變量名稱及變量值之大小寫是有區別的(case sensitive)。

如下是一些變量設定時常見的錯誤:
        A= B        :不能有 IFS
        1A=B        :不能以數字開頭
        $A=B        :名稱不能有 $
        a=B                :這跟 a=b 是不同的(這不是錯誤,提醒 windows 的使用者要特別注意)
如下則是可以接受的設定:
        A=" B"        :IFS 被關閉了 (請參考前面的 quoting 章節)
        A1=B        :並非以數字開頭
        A=$B        :$ 可用在變量值內
        This_Is_A_Long_Name=b        :可用 _ 連接較長的名稱或值,且大小寫有別。

*變量替換(substitution)*
Shell 之所以強大,其中的一個因素是它可以在命令行中對變量作替換(substitution)處理。
在命令行中使用者可以使用 $ 符號加上變量名稱(除了在用 = 號定義變量名稱之外),
將變量值給替換出來,然後再重新組建命令行。
比方:
 

  1.         $ A=ls
     
  2.         $ B=la
     
  3.         $ C=/tmp
     
  4.         $ $A -$B $C
复制代码


        (注意:以上命令行的第一個 $ 是 shell prompt ,並不在命令行之內。)
必需強調的是,我們所提的變量替換,只發生在 command line 上面。(是的,讓我們再回到 command line 吧﹗)
仔細分析最後那行 command line ,不難發現在被執行之前(在輸入 CR 字符之前),
$ 符號會對每一個變量作替換處理(將變量值替換出來再重組命令行),最後會得出如下命令行:
 

  1.         ls -la /tmp
复制代码


還記得第二章我請大家"務必理解"的那兩句嗎?若你忘了,那我這裡再重貼一遍:
 

若從技術細節來看,shell 會依據 IFS(Internal Field Seperator) 將 command line 所輸入的文字給拆解為"字段"(word)。
然後再針對特殊字符(meta)先作處理,最後再重組整行 command line 。


這裡的 $ 就是 command line 中最經典的 meta 之一了,就是作變量替換的﹗
在日常的 shell 操作中,我們常會使用 echo 命令來查看特定變量的值,例如:
 

  1.         $ echo $A -$B $C
复制代码


我們已學過, echo 命令只單純將其 argument 送至"標準輸出"(STDOUT,通常是我們的熒幕)。
所以上面的命令會在熒幕上得到如下結果:
 

  1.         ls -la /tmp
复制代码


這是由於 echo 命令在執行時,會先將 $A(ls)、$B(la)、跟 $C(/tmp) 給替換出來的結果。

利用 shell 對變量的替換處理能力,我們在設定變量時就更為靈活了:
        A=B
        B=$A
這樣,B 的變量值就可繼承 A 變量"當時"的變量值了。
不過,不要以"數學羅輯"來套用變量的設定,比方說:
        A=B
        B=C
這樣並不會讓 A 的變量值變成 C 。再如:
        A=B
        B=$A
        A=C
同樣也不會讓 B 的值換成 C 。
上面是單純定義了兩個不同名稱的變量:A 與  B ,它們的值分別是 B 與  C 。
若變量被重復定義的話,則原有舊值將被新值所取代。(這不正是"可變的量"嗎?  ^_^)
當我們在設定變量的時侯,請記著這點:
        * 用一個名稱儲存一個數值
僅此而已。

此外,我們也可利用命令行的變量替換能力來"擴充"(append)變量值:
        A=B:C:D
        A=$A:E
這樣,第一行我們設定 A 的值為 "B:C:D",然後,第二行再將值擴充為 "B:C:D:E" 。
上面的擴充範例,我們使用區隔符號( : )來達到擴充目的,
要是沒有區隔符號的話,如下是有問題的:
        A=BCD
        A=$AE
因為第二次是將 A 的值繼承 $AE 的提換結果,而非 $A 再加 E ﹗
要解決此問題,我們可用更嚴謹的替換處理:
        A=BCD
        A=${A}E
上例中,我們使用 {} 將變量名稱的範圍給明確定義出來,
如此一來,我們就可以將 A 的變量值從 BCD 給擴充為 BCDE 。

(提示:關於 ${name} 事實上還可做到更多的變量處理能力,這些均屬於比較進階的變量處理,
現階段暫時不介紹了,請大家自行參考資料。如 CU 的貼子:
http://www.chinaunix.net/forum/viewtopic.php?t=201843
)

* export *

嚴格來說,我們在當前 shell 中所定義的變量,均屬於"本地變量"(local variable),
只有經過 export 命令的"輸出"處理,才能成為環境變量(environment variable):
 

  1.         $ A=B
     
  2.         $ export A
复制代码


或:
 

  1.         $ export A=B
复制代码


經過 export 輸出處理之後,變量 A 就能成為一個環境變量供其後的命令使用。
在使用 export  的時侯,請別忘記 shell 在命令行對變量的"替換"(substitution)處理,
比方說:
 

  1.         $ A=B
     
  2.         $ B=C
     
  3.         $ export $A
复制代码


上面的命令並未將 A 輸出為環境變量,而是將 B 作輸出,
這是因為在這個命令行中,$A 會首先被替換為 B, 然後再"塞回"作 export 的參數。

要理解這個 export ,事實上需要從 process 的角度來理解才能透徹。
我將於下一章為大家說明 process 的觀念,敬請留意。

*取消變量*

要取消一個變量,在 bash 中可使用 unset 命令來處理:
 

  1.         unset A
复制代码


與 export 一樣,unset 命令行也同樣會作變量替換(這其實就是 shell 的功能之一),
因此:
 

  1.         $ A=B
     
  2.         $ B=C
     
  3.         $ unset $A
复制代码


事實上所取消的變量是 B 而不是 A 。

此外,變量一旦經過 unset 取消之後,其結果是將整個變量拿掉,而不僅是取消其變量值。
如下兩行其實是很不一樣的:
 

  1.         $ A=
     
  2.         $ unset A
复制代码


第一行只是將變量 A 設定為"空值"(null  value),但第二行則讓變量 A 不在存在。
雖然用眼睛來看,這兩種變量狀態在如下命令結果中都是一樣的:
 

  1.         $ A=
     
  2.         $ echo $A
     

  3.  
  4.         $ unset A
     
  5.         $ echo $A
     
  6.        
复制代码


請學員務必能識別 null value 與 unset 的本質區別,這在一些進階的變量處理上是很嚴格的。
比方說:
 

  1.         $ str=                # 設為 null
     
  2.         $ var=${str=expr}        # 定義 var
     
  3.         $ echo $var
     
  4.        
     
  5.         $ echo $str
     
  6.        
     
  7.         $ unset str        # 取消
     
  8.         $ var=${str=expr}        # 定義 var
     
  9.         $ echo $var
     
  10.         expr
     
  11.         $ echo $str
     
  12.         expr
复制代码


聰明的讀者(yes, you!),稍加思考的話,
應該不難發現為何同樣的 var=${str=expr} 在 null 與 unset 之下的不同吧?
若你看不出來,那可能是如下原因之一:
a. 你太笨了
b. 不了解  var=${str=expr}        這個進階處理
c. 對本篇說明還沒來得及消化吸收
e. 我講得不好
不知,你選哪個呢?....  ^_^

嗯... 好吧,我就解釋一下 var=${str=expr} :

首先,var=$str 這個大家都可理解吧。
而接下來的思考方向是,究竟 $str 這個變量是如下哪一種情況呢:
1) unset
2) null
3) not null

1) 假如是 unset ,那麼 var=${str=expr} 的結果將是:
var=expr
str=expr

2) 假如是 null ,那 var=${str=expr} 的結果是:
var=
str=

3) 假如是 not null (比方為 xyz ),那 var=${str=expr} 之結果是:
var=xyz
str=xyz

 

接下來,再來看看 var=${str:=expr} 好了:
1) $str 為 not set :
var=expr
str=expr

2) $str 為 null :
var=expr
str=expr

3) $str 為 not null (str=xyz):
var=xyz
str=xyz

最後比教一下 ${str=expr} 與 ${str:=expr} :
* 兩者在 not set 與 not null 都一至
* 但當 null 值時,前者會將 $var 與 $str 都設為 null ,但後者則設為 expr

從這個再延伸出其他類比,不防請大家"實作"觀查一下有何不同?
var=${str-expr} vs var=${str:-expr}
var=${str+expr} vs var=${str:+expr}
var=${str?expr} vs var=${strexpr}

hey you: 別偷懶﹗快做做看... hurry up!