linux命令詳解——eval

shell中的eval

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

語  法:eval [參數]

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

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

 

1.eval命令將會首先掃描命令行進行全部的替換,憨厚再執行命令。該命令使用於那些一次掃描沒法實現其功能的變量。該命令對變量進行兩次掃描。這些須要進行兩次掃描的變量有時候被稱爲複雜變量。linux

2.eval也能夠用於回顯簡單變量,不必定時複雜變量。程序員

NAME=ZONEshell

eval echo $NAME等價於echo $NAME數據庫

3.兩次掃描bash

test.txt內容:hello shell world!函數

myfile="cat test.txt"測試

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

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

從(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_r(eval_r("code")),來執行(執行動態生成的code的返回),而因爲shell 中 eval 將後面的eval命令簡單看成命令字符串執行,失去了嵌套做用,嵌套被命令替換取代。

擴展閱讀:eval命令使用示例詳解 資料整理www.linuxso.com

eval的做用是再次執行命令行處理,也就是說,對一個命令行,執行兩次命令行處理。這個命令要用好,就要費必定的功夫。我舉兩個例子,拋磚引玉。 一、例子1:用eval技巧實現shell的控制結構for 用eval技巧實現shell的控制結構for。 [root@home root]# cat myscript1 QUOTE: #!/bin/sh evalit(){         if [ $cnt = 1 ];then                 eval $@                 return         elee                 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程序員的必修之技。

相關文章
相關標籤/搜索