什么是Shell腳本示例看個(gè)例子吧: #!/bin/shcd ~mkdir shell_tutcd shell_tutfor ((i=0; i<10; i++)); do touch test_$i.txtdone 示例解釋
cd, mkdir, touch都是系統(tǒng)自帶的程序,一般在/bin或者/usr/bin目錄下,。for, do, done是sh腳本語(yǔ)言的關(guān)鍵字,。 shell和shell腳本的概念shell是指一種應(yīng)用程序,這個(gè)應(yīng)用程序提供了一個(gè)界面,,用戶通過(guò)這個(gè)界面訪問(wèn)操作系統(tǒng)內(nèi)核的服務(wù),。Ken Thompson的sh是第一種Unix Shell,Windows Explorer是一個(gè)典型的圖形界面Shell,。 shell腳本(shell script),,是一種為shell編寫(xiě)的腳本程序。業(yè)界所說(shuō)的shell通常都是指shell腳本,,但讀者朋友要知道,shell和shell script是兩個(gè)不同的概念,。由于習(xí)慣的原因,,簡(jiǎn)潔起見(jiàn),,本文出現(xiàn)的“shell編程”都是指shell腳本編程,,不是指開(kāi)發(fā)shell自身(如Windows Explorer擴(kuò)展開(kāi)發(fā)),。 環(huán)境shell編程跟java、php編程一樣,,只要有一個(gè)能編寫(xiě)代碼的文本編輯器和一個(gè)能解釋執(zhí)行的腳本解釋器就可以了。 OS當(dāng)前主流的操作系統(tǒng)都支持shell編程,,本文檔所述的shell編程是指Linux下的shell,,講的基本都是POSIX標(biāo)準(zhǔn)下的功能,,所以,,也適用于Unix及BSD(如Mac OS)。 LinuxLinux默認(rèn)安裝就帶了shell解釋器,。 Mac OSMac OS不僅帶了sh,、bash這兩個(gè)最基礎(chǔ)的解釋器,,還內(nèi)置了ksh、csh,、zsh等不常用的解釋器,。 Windows上的模擬器windows出廠時(shí)沒(méi)有內(nèi)置shell解釋器,,需要自行安裝,,為了同時(shí)能用grep, awk, curl等工具,最好裝一個(gè)cygwin或者mingw來(lái)模擬linux環(huán)境,。 腳本解釋器sh即Bourne shell,,POSIX(Portable Operating System Interface)標(biāo)準(zhǔn)的shell解釋器,它的二進(jìn)制文件路徑通常是/bin/sh,,由Bell Labs開(kāi)發(fā)。 本文講的是sh,,如果你使用其它語(yǔ)言用作shell編程,請(qǐng)自行參考相應(yīng)語(yǔ)言的文檔,。 bashBash是Bourne shell的替代品,,屬GNU Project,,二進(jìn)制文件路徑通常是/bin/bash,。業(yè)界通?;煊胋ash、sh,、和shell,,比如你會(huì)經(jīng)常在招聘運(yùn)維工程師的文案中見(jiàn)到:熟悉Linux Bash編程,,精通Shell編程。 在CentOS里,,/bin/sh是一個(gè)指向/bin/bash的符號(hào)鏈接: [root@centosraw ~]# ls -l /bin/*sh-rwxr-xr-x. 1 root root 903272 Feb 22 05:09 /bin/bash-rwxr-xr-x. 1 root root 106216 Oct 17 2012 /bin/dashlrwxrwxrwx. 1 root root 4 Mar 22 10:22 /bin/sh -> bash 但在Mac OS上不是,/bin/sh和/bin/bash是兩個(gè)不同的文件,,盡管它們的大小只相差100字節(jié)左右: iMac:~ wuxiao$ ls -l /bin/*sh-r-xr-xr-x 1 root wheel 1371648 6 Nov 16:52 /bin/bash-rwxr-xr-x 2 root wheel 772992 6 Nov 16:52 /bin/csh-r-xr-xr-x 1 root wheel 2180736 6 Nov 16:52 /bin/ksh-r-xr-xr-x 1 root wheel 1371712 6 Nov 16:52 /bin/sh-rwxr-xr-x 2 root wheel 772992 6 Nov 16:52 /bin/tcsh-rwxr-xr-x 1 root wheel 1103984 6 Nov 16:52 /bin/zsh 高級(jí)編程語(yǔ)言理論上講,,只要一門(mén)語(yǔ)言提供了解釋器(而不僅是編譯器),,這門(mén)語(yǔ)言就可以勝任腳本編程,,常見(jiàn)的解釋型語(yǔ)言都是可以用作腳本編程的,如:Perl,、Tcl,、Python,、PHP,、Ruby,。Perl是最老牌的腳本編程語(yǔ)言了,,Python這些年也成了一些linux發(fā)行版的預(yù)置解釋器,。 編譯型語(yǔ)言,只要有解釋器,,也可以用作腳本編程,,如C shell是內(nèi)置的(/bin/csh),Java有第三方解釋器Jshell,,Ada有收費(fèi)的解釋器AdaScript,。 如下是一個(gè)PHP Shell Script示例(假設(shè)文件名叫test.php): #!/usr/bin/phpfor ($i=0; $i < 10; $i++) echo $i . '\n'; 執(zhí)行: /usr/bin/php test.php 或者: chmod +x test.php./test.php 如何選擇shell編程語(yǔ)言熟悉 vs 陌生如果你已經(jīng)掌握了一門(mén)編程語(yǔ)言(如PHP、Python,、Java,、JavaScript),建議你就直接使用這門(mén)語(yǔ)言編寫(xiě)腳本程序,,雖然某些地方會(huì)有點(diǎn)啰嗦,,但你能利用在這門(mén)語(yǔ)言領(lǐng)域里的經(jīng)驗(yàn)(單元測(cè)試、單步調(diào)試,、IDE、第三方類(lèi)庫(kù)),。 新增的學(xué)習(xí)成本很小,,只要學(xué)會(huì)怎么使用shell解釋器(Jshell、AdaScript)就可以了,。 簡(jiǎn)單 vs 高級(jí)如果你覺(jué)得自己熟悉的語(yǔ)言(如Java,、C)寫(xiě)shell腳本實(shí)在太啰嗦,你只是想做一些備份文件,、安裝軟件,、下載數(shù)據(jù)之類(lèi)的事情,學(xué)著使用sh,,bash會(huì)是一個(gè)好主意,。 shell只定義了一個(gè)非常簡(jiǎn)單的編程語(yǔ)言,所以,,如果你的腳本程序復(fù)雜度較高,,或者要操作的數(shù)據(jù)結(jié)構(gòu)比較復(fù)雜,那么還是應(yīng)該使用Python,、Perl這樣的腳本語(yǔ)言,,或者是你本來(lái)就已經(jīng)很擅長(zhǎng)的高級(jí)語(yǔ)言,。因?yàn)閟h和bash在這方面很弱,比如說(shuō):
環(huán)境兼容性如果你的腳本是提供給別的用戶使用,,使用sh或者bash,你的腳本將具有最好的環(huán)境兼容性,,perl很早就是linux標(biāo)配了,,python這些年也成了一些linux發(fā)行版的標(biāo)配,至于mac os,,它默認(rèn)安裝了perl,、python、ruby,、php,、java等主流編程語(yǔ)言。 第一個(gè)shell腳本編寫(xiě)打開(kāi)文本編輯器,,新建一個(gè)文件,,擴(kuò)展名為sh(sh代表shell),擴(kuò)展名并不影響腳本執(zhí)行,,見(jiàn)名知意就好,,如果你用php寫(xiě)shell 腳本,擴(kuò)展名就用php好了,。 輸入一些代碼,,第一行一般是這樣: #!/bin/bash#!/usr/bin/php “#!”是一個(gè)約定的標(biāo)記,它告訴系統(tǒng)這個(gè)腳本需要什么解釋器來(lái)執(zhí)行,。 運(yùn)行運(yùn)行Shell腳本有兩種方法: 作為可執(zhí)行程序chmod +x test.sh./test.sh 注意,,一定要寫(xiě)成./test.sh,而不是test.sh,,運(yùn)行其它二進(jìn)制的程序也一樣,,直接寫(xiě)test.sh,linux系統(tǒng)會(huì)去PATH里尋找有沒(méi)有叫test.sh的,,而只有/bin, /sbin, /usr/bin,,/usr/sbin等在PATH里,你的當(dāng)前目錄通常不在PATH里,,所以寫(xiě)成test.sh是會(huì)找不到命令的,,要用./test.sh告訴系統(tǒng)說(shuō),,就在當(dāng)前目錄找。 通過(guò)這種方式運(yùn)行bash腳本,,第一行一定要寫(xiě)對(duì),,好讓系統(tǒng)查找到正確的解釋器。 這里的”系統(tǒng)”,,其實(shí)就是shell這個(gè)應(yīng)用程序(想象一下Windows Explorer),,但我故意寫(xiě)成系統(tǒng),是方便理解,,既然這個(gè)系統(tǒng)就是指shell,,那么一個(gè)使用/bin/sh作為解釋器的腳本是不是可以省去第一行呢?是的,。 作為解釋器參數(shù)這種運(yùn)行方式是,,直接運(yùn)行解釋器,其參數(shù)就是shell腳本的文件名,,如: /bin/sh test.sh/bin/php test.php 這種方式運(yùn)行的腳本,,不需要在第一行指定解釋器信息,寫(xiě)了也沒(méi)用,。 變量定義變量定義變量時(shí),,變量名不加美元符號(hào)($),如: your_name='qinjx' 注意,,變量名和等號(hào)之間不能有空格,,這可能和你熟悉的所有編程語(yǔ)言都不一樣。 除了顯式地直接賦值,,還可以用語(yǔ)句給變量賦值,,如: for file in `ls /etc` 使用變量使用一個(gè)定義過(guò)的變量,只要在變量名前面加美元符號(hào)即可,,如: your_name='qinjx'echo $your_nameecho ${your_name} 變量名外面的花括號(hào)是可選的,加不加都行,,加花括號(hào)是為了幫助解釋器識(shí)別變量的邊界,,比如下面這種情況: for skill in Ada Coffe Action Java; do echo 'I am good at ${skill}Script'done 如果不給skill變量加花括號(hào),寫(xiě)成echo “I am good at $skillScript”,,解釋器就會(huì)把$skillScript當(dāng)成一個(gè)變量(其值為空),,代碼執(zhí)行結(jié)果就不是我們期望的樣子了。 推薦給所有變量加上花括號(hào),,這是個(gè)好的編程習(xí)慣,。IntelliJ IDEA編寫(xiě)shell script時(shí),IDE就會(huì)提示加花括號(hào),。 重定義變量已定義的變量,,可以被重新定義,,如: your_name='qinjx'echo $your_nameyour_name='alibaba'echo $your_name 這樣寫(xiě)是合法的,但注意,,第二次賦值的時(shí)候不能寫(xiě)$your_name=”alibaba”,,使用變量的時(shí)候才加美元符。 注釋以“#”開(kāi)頭的行就是注釋,,會(huì)被解釋器忽略,。 多行注釋sh里沒(méi)有多行注釋,只能每一行加一個(gè)#號(hào),。就像這樣: #--------------------------------------------# 這是一個(gè)自動(dòng)打ipa的腳本,,基于webfrogs的ipa-build書(shū)寫(xiě):https://github.com/webfrogs/xcode_shell/blob/master/ipa-build# 功能:自動(dòng)為etao ios app打包,產(chǎn)出物為14個(gè)渠道的ipa包# 特色:全自動(dòng)打包,,不需要輸入任何參數(shù)#--------------------------------------------##### 用戶配置區(qū) 開(kāi)始 ######## 項(xiàng)目根目錄,,推薦將此腳本放在項(xiàng)目的根目錄,這里就不用改了# 應(yīng)用名,,確保和Xcode里Product下的target_name.app名字一致###### 用戶配置區(qū) 結(jié)束 ##### 如果在開(kāi)發(fā)過(guò)程中,,遇到大段的代碼需要臨時(shí)注釋起來(lái),過(guò)一會(huì)兒又取消注釋,,怎么辦呢,?每一行加個(gè)#符號(hào)太費(fèi)力了,可以把這一段要注釋的代碼用一對(duì)花括號(hào)括起來(lái),,定義成一個(gè)函數(shù),,沒(méi)有地方調(diào)用這個(gè)函數(shù),這塊代碼就不會(huì)執(zhí)行,,達(dá)到了和注釋一樣的效果,。 字符串字符串是shell編程中最常用最有用的數(shù)據(jù)類(lèi)型(除了數(shù)字和字符串,也沒(méi)啥其它類(lèi)型好用了,,哈哈),,字符串可以用單引號(hào),也可以用雙引號(hào),,也可以不用引號(hào),。單雙引號(hào)的區(qū)別跟PHP類(lèi)似。 單引號(hào)str='this is a string' 單引號(hào)字符串的限制:
雙引號(hào)your_name='qinjx'str='Hello, I know your are \'$your_name\'! \n'
字符串操作拼接字符串your_name='qinjx'greeting='hello, '$your_name' !'greeting_1='hello, ${your_name} !'echo $greeting $greeting_1 獲取字符串長(zhǎng)度:string='abcd'echo ${#string} #輸出:4 提取子字符串string='alibaba is a great company'echo ${string:1:4} #輸出:liba 查找子字符串string='alibaba is a great company'echo `expr index '$string' is`#輸出:8,,這個(gè)語(yǔ)句的意思是:找出單詞is在這名話中的位置 數(shù)組管道條件判斷流程控制和Java、PHP等語(yǔ)言不一樣,,sh的流程控制不可為空,,如: if (isset($_GET['q'])) { search(q);}else { //do nothing} 在sh/bash里可不能這么寫(xiě),如果else分支沒(méi)有語(yǔ)句執(zhí)行,就不要寫(xiě)這個(gè)else,。 還要注意,,sh里的if [ $foo -eq 0 ],這個(gè)方括號(hào)跟Java/PHP里if后面的圓括號(hào)大不相同,,它是一個(gè)可執(zhí)行程序(和cd, ls, grep一樣),,想不到吧?在CentOS上,,它在/usr/bin目錄下: ll /usr/bin/[-rwxr-xr-x. 1 root root 33408 6月 22 2012 /usr/bin/[ 正因?yàn)榉嚼ㄌ?hào)在這里是一個(gè)可執(zhí)行程序,,方括號(hào)后面必須加空格,不能寫(xiě)成if [$foo -eq 0] if elseifif conditionthen command1 command2 ... commandN fi 寫(xiě)成一行(適用于終端命令提示符): if `ps -ef | grep ssh`; then echo hello; fi 末尾的fi就是if倒過(guò)來(lái)拼寫(xiě),,后面還會(huì)遇到類(lèi)似的 if elseif conditionthen command1 command2 ... commandNelse commandfi if else-if elseif condition1then command1elif condition2 command2else commandNfi for whilefor在開(kāi)篇的示例里演示過(guò)了: for var in item1 item2 ... itemNdo command1 command2 ... commandNdone 寫(xiě)成一行: for var in item1 item2 ... itemN; do command1; command2… done; C風(fēng)格的forfor (( EXP1; EXP2; EXP3 ))do command1 command2 command3done whilewhile conditiondo commanddone 無(wú)限循環(huán)while :do commanddone 或者 while truedo commanddone 或者 for (( ; ; )) untiluntil conditiondo commanddone casecase '${opt}' in 'Install-Puppet-Server' ) install_master $1 exit ;; 'Install-Puppet-Client' ) install_client $1 exit ;; 'Config-Puppet-Server' ) config_puppet_master exit ;; 'Config-Puppet-Client' ) config_puppet_client exit ;; 'Exit' ) exit ;; * ) echo 'Bad option, please choose again'esac case的語(yǔ)法和C family語(yǔ)言差別很大,,它需要一個(gè)esac(就是case反過(guò)來(lái))作為結(jié)束標(biāo)記,每個(gè)case分支用右圓括號(hào),,用兩個(gè)分號(hào)表示break 函數(shù)定義調(diào)用文件包含可以使用source和.關(guān)鍵字,,如: source ./function.sh. ./function.sh 在bash里,source和.是等效的,,他們都是讀入function.sh的內(nèi)容并執(zhí)行其內(nèi)容(類(lèi)似PHP里的include),,為了更好的可移植性,推薦使用第二種寫(xiě)法,。 包含一個(gè)文件和執(zhí)行一個(gè)文件一樣,,也要寫(xiě)這個(gè)文件的路徑,不能光寫(xiě)文件名,,比如上述例子中: . ./function.sh 不可以寫(xiě)作: . function.sh 如果function.sh是用戶傳入的參數(shù),,如何獲得它的絕對(duì)路徑呢?方法是: real_path=`readlink -f $1`#$1是用戶輸入的參數(shù),,如function.sh. $real_path 用戶輸入執(zhí)行腳本時(shí)傳入腳本運(yùn)行中輸入select菜單stdin和stdout常用的命令sh腳本結(jié)合系統(tǒng)命令便有了強(qiáng)大的威力,,在字符處理領(lǐng)域,有g(shù)rep,、awk,、sed三劍客,grep負(fù)責(zé)找出特定的行,,awk能將行拆分成多個(gè)字段,,sed則可以實(shí)現(xiàn)更新插入刪除等寫(xiě)操作。 ps查看進(jìn)程列表 grep排除grep自身查找與target相鄰的結(jié)果awksed插入替換刪除xargscurl |
|