shell 中的 eval

來源:互聯網
上載者:User

功能說明:重新運算求出參數的內容。

語  法:eval [參數]

補充說明:eval可讀取一連串的參數,然後再依參數本身的特性來執行。

參  數:參數不限數目,彼此之間用分號分開。

 

1.eval命令將會首先掃描命令列進行所有的替換,憨厚再執行命令。該命令使用於那些一次掃描無法實現其功能的變數。該命令對變數進行兩次掃描。這些需要進行兩次掃描的變數有時候被稱為複雜變數。

2.eval也可以用於回顯簡單變數,不一定時複雜變數。

NAME=ZONE

eval echo $NAME等價於echo $NAME

3.兩次掃描

test.txt內容:hello shell world!

myfile="cat test.txt"

(1)echo $myfile  #result:cat test.txt

(2)eval echo $myfile  #result:hello shell world!

從(2)可以知道第一次掃描進行了變數替換,第二次掃描執行了該字串中所包含的命令

4.獲得最後一個參數

echo "Last argument is $(eval echo \$$#)"

echo "Last argument is $(eval echo $#)"

 

shell 也提供了 eval 命令,如同熟悉的其他指令碼語言,會將它的參數做為命令執行,初看會疑惑為什麼shell要提供兩種動態執行命令字串的機制,但是經過仔細分析,才發現shell的eval同其他語言有很大區別。

1.shell 中的函數雖然可以通過return 返回,但是這裡的return 相當於 exit,只能是個狀態值用於測試,而不能像其它語言一樣返回複雜的結果,其處理結果只能通過輸出到標準輸出經過 `` ,$()取得。

2.shell 中的 eval

    2.1 不能獲得函數處理結果 ,如1所說,所有命令,函數的處理結果只能通過 ``來獲得,那麼其它語言中利用eval來獲得動態產生代碼執行後的輸出變得不可能。

    2.2 eval 嵌套無意義 ,在其他語言中可以通過 eval(eval("code")),來執行(執行動態產生的code的返回),而由於shell 中 eval 將後面的eval命令簡單當作命令字串執行,失去了嵌套作用,嵌套被命令替換取代。

擴充閱讀:eval命令使用樣本詳解 資料整理www.linuxso.com

eval的作用是再次執行命令列處理,也就是說,對一個命令列,執行兩次命令列處理。這個命令要用好,就要費一定的功夫。我舉兩個例子,拋磚引玉。

1、例子1:用eval技巧實現shell的控制結構for

用eval技巧實現shell的控制結構for。

[root@home root]# cat myscript1

QUOTE:
#!/bin/sh
evalit(){
        if [ $cnt = 1 ];then
                eval $@
                return
        else
                let cnt="cnt-1"
                evalit $@
        fi
        eval $@
}
cnt=$1
echo $cnt | egrep "^[1-9][0-9]*$" >/dev/null
if [ $? -eq 0 ]; then
        shift
        evalit $@
else
        echo 'ERROR!!! Check your input!'
fi
[root@home root]# ./myscript1 3 hostname
home
home
home
[root@home root]# ./myscript1 5 id |cut -f1 -d' '
uid=0(root)
uid=0(root)
uid=0(root)
uid=0(root)
uid=0(root)
注意:bash裡有兩個很特殊的變數,它們儲存了參數列表。

$*,儲存了以$IFS指定的分割符所分割的字串組。
$@,原樣儲存了參數列表,也就是"$1""$2"...

這裡我使用了函數遞迴以及eval實現了for結構。
當執行eval $@時,它經曆了步驟如下:
第1步,分割成eval $@
第6步,擴充$@為hostname
第11步,找到內建命令eval
重複一次命令列處理,第11步,找到hostname命令,執行。

注意:也許有人想當然地認為,何必用eval呢?直接$@來執行命令就可以了嘛。

例子2:一個典型錯誤的例子

錯誤!這裡給個典型的例子大家看看。

[root@home root]# a="id | cut -f1 -d' '"
[root@home root]# $a
id:無效選項 #  f
請嘗試執行‘id # help’來擷取更多資訊。
[root@home root]# eval $a
uid=0(root)

如果命令列複雜的話(包括管道或者其他字元),直接執行$a字串的內容就會出錯。分析如下。
$a的處理位於第6步──參數擴充,也就是說,跳過了管道分析,於是"|", "cut", "-f1", "-d"都變成了id命令的參數,當然就出錯啦。
但使用了eval,它把第一遍命令列處理所得的"id", "|", "cut", "-f1", "-d"這些字串再次進行命令列處理,這次就能正確分析其中的管道了。

總而言之:要保證你的命令或指令碼設計能正確通過命令列處理,跳過任意一步,都可能造成意料外的錯誤!

例子3:設定系統的ls色彩顯示

eval $(dircolors -b /etc/dircolors)

eval語句通知shell接受eval參數,並再次通過命令列處理的所有步驟運行它們。
它使你可以編寫指令碼隨意建立命令字串,然後把它們傳遞給shell執行;
$()是命令替換,返回命令的輸出字串。
其中dircolors命令根據/etc/dircolors設定檔產生設定環境變數LS_COLORS的bash代碼,內容如下

[root@localhost root]# dircolors -b > tmp
[root@localhost root]# cat tmp
LS_COLORS='no=00:fi=00:di=01;34:ln=01; ......
export LS_COLORS
#這裡我沒有指定設定檔,所以dircolors按預置資料庫產生代碼。
其輸出被eval命令傳遞給shell執行。

eval是對Bash Shell命令列處理規則的靈活應用,進而構造"智能"命令實現複雜的功能。
上面提及的命令是eval其中一個很普通的應用,它重複了1次命令列參數傳遞過程,純粹地執行命令的命令。
其實它是bash的痛點,是進階bash程式員的必修之技。

參考資料:http://www.linuxso.com/command/eval.html

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.