在 Linux 中使用變數

2019-09-15 10:51:00

變數通常看起來像 $var 這樣,但它們也有 $1、$*、$? 和 $$ 這種形式。讓我們來看看所有這些 $ 值可以告訴你什麼。

有許多重要的值都儲存在 Linux 系統中,我們稱為“變數”,但實際上變數有幾種型別,並且一些有趣的命令可以幫助你使用它們。在上一篇文章中,我們研究了以及它們定義在何處。在本文中,我們來看一看在命令列和指令碼中使用的變數。

使用者變數

雖然在命令列中設定變數非常容易,但是有一些有趣的技巧。要設定變數,你只需這樣做:

$ myvar=11$ myvar2="eleven"

要顯示這些值,只需這樣做:

$ echo $myvar11$ echo $myvar2eleven

你也可以使用這些變數。例如,要遞增一個數位變數,使用以下任意一個命令:

$ myvar=$((myvar+1))$ echo $myvar12$ ((myvar=myvar+1))$ echo $myvar13$ ((myvar+=1))$ echo $myvar14$ ((myvar++))$ echo $myvar15$ let "myvar=myvar+1"$ echo $myvar16$ let "myvar+=1"$ echo $myvar17$ let "myvar++"$ echo $myvar18

使用其中的一些,你可以增加一個變數的值。例如:

$ myvar0=0$ ((myvar0++))$ echo $myvar01$ ((myvar0+=10))$ echo $myvar011

通過這些選項,你可能會發現它們是容易記憶、使用方便的。

你也可以刪除一個變數 – 這意味著沒有定義它。

$ unset myvar$ echo $myvar

另一個有趣的選項是,你可以設定一個變數並將其設為唯讀。換句話說,變數一旦設定為唯讀,它的值就不能改變(除非一些非常複雜的命令列魔法才可以)。這意味著你也不能刪除它。

$ readonly myvar3=1$ echo $myvar31$ ((myvar3++))-bash: myvar3: readonly variable$ unset myvar3-bash: unset: myvar3: cannot unset: readonly variable

你可以使用這些設定和遞增選項中來賦值和操作指令碼中的變數,但也有一些非常有用的內部變數可以用於在指令碼中。注意,你無法重新賦值或增加它們的值。

內部變數

在指令碼中可以使用很多變數來計算引數並顯示有關指令碼本身的資訊。

  • $1$2$3 等表示指令碼的第一個、第二個、第三個等引數。
  • $# 表示引數的數量。
  • $* 表示所有引數。
  • $0 表示指令碼的名稱。
  • $? 表示先前執行的命令的返回碼(0 代表成功)。
  • $$ 顯示指令碼的進程 ID。
  • $PPID 顯示 shell 的進程 ID(指令碼的父進程)。

其中一些變數也適用於命令列,但顯示相關資訊:

  • $0 顯示你正在使用的 shell 的名稱(例如,-bash)。
  • $$ 顯示 shell 的進程 ID。
  • $PPID 顯示 shell 的父進程的進程 ID(對我來說,是 sshd)。

為了檢視它們的結果,如果我們將所有這些變數都放入一個指令碼中,比如:

#!/bin/bashecho $0echo $1echo $2echo $#echo $*echo $?echo $$echo $PPID

當我們呼叫這個指令碼時,我們會看到如下內容:

$ tryme one two three/home/shs/bin/tryme     <== 指令碼名稱one                     <== 第一個引數two                     <== 第二個引數3                       <== 引數的個數one two three           <== 所有的引數0                       <== 上一條 echo 命令的返回碼10410                   <== 指令碼的進程 ID10109                   <== 父進程 ID

如果我們在指令碼執行完畢後檢查 shell 的進程 ID,我們可以看到它與指令碼中顯示的 PPID 相匹配:

$ echo $$10109                   <== shell 的進程 ID

當然,比起簡單地顯示它們的值,更有用的方式是使用它們。我們來看一看它們可能的用處。

檢查是否已提供引數:

if [ $# == 0 ]; then    echo "$0 filename"    exit 1fi

檢查特定進程是否正在執行:

ps -ef | grep apache2 > /dev/nullif [ $? != 0 ]; then    echo Apache is not running    exitfi

在嘗試存取檔案之前驗證檔案是否存在:

if [ $# -lt 2 ]; then    echo "Usage: $0 lines filename"    exit 1fiif [ ! -f $2 ]; then    echo "Error: File $2 not found"    exit 2else    head -$1 $2fi

在下面的小指令碼中,我們檢查是否提供了正確數量的引數、第一個引數是否為數位,以及第二個引數代表的檔案是否存在。

#!/bin/bashif [ $# -lt 2 ]; then    echo "Usage: $0 lines filename"    exit 1fiif [[ $1 != [0-9]* ]]; then    echo "Error: $1 is not numeric"    exit 2fiif [ ! -f $2 ]; then    echo "Error: File $2 not found"    exit 3else    echo top of file    head -$1 $2fi

重新命名變數

在編寫複雜的指令碼時,為指令碼的引數指定名稱通常很有用,而不是繼續將它們稱為 $1$2 等。等到第 35 行,閱讀你指令碼的人可能已經忘了 $2 表示什麼。如果你將一個重要引數的值賦給 $filename$numlines,那麼他就不容易忘記。

#!/bin/bashif [ $# -lt 2 ]; then    echo "Usage: $0 lines filename"    exit 1else    numlines=$1    filename=$2fiif [[ $numlines != [0-9]* ]]; then    echo "Error: $numlines is not numeric"    exit 2fiif [ ! -f $ filename]; then    echo "Error: File $filename not found"    exit 3else    echo top of file    head -$numlines $filenamefi

當然,這個範例指令碼只是執行 head 命令來顯示檔案中的前 x 行,但它的目的是顯示如何在指令碼中使用內部引數來幫助確保指令碼執行良好,或在失敗時清晰地知道失敗原因。