shell Unix系統中常見的shell有:sh、csh、ksh、tcsh、bash 使用命令: vim /etc/shells 可以查看當前系統支持的bash shell執行腳本 shell執行腳本是一門解釋性語言、批量化處理語言,大大的節省了工作成本 shell腳本第一行必須以 #!開頭,它表 ...
shell
- Shell就是一個命令行解釋器,它的作用是解釋執行用戶的命令,用戶輸入一條命令,Shell就解釋執行一條,這種方式稱為互動式(Interactive)。
- Shell還有一種執行命令的方式稱為批處理(Batch),用戶事先寫一 個Shell腳本(Script),其中有很多條命令,讓Shell一次把這些命令執行完,而不必一條一條地敲 命令。Shell腳本和編程語言很相似,也有變數和流程式控制制語句,包括迴圈和分支。但Shell腳本是解釋執行的,不需 要編譯,Shell程式從腳本中一行一行讀取並執行這些命令,相當於一個用戶把腳本中的命令一行一 行敲到Shell提示符下執行。作為程式設計語言,它雖然不是 Linux系統內核的一部分,但它調用了系統內核的大部分功能來執行程式、創建文檔並以並行的方式協調各個程式的運行。
Unix系統中常見的shell有:sh、csh、ksh、tcsh、bash
使用命令: vim /etc/shells 可以查看當前系統支持的bash
shell執行腳本
shell執行腳本是一門解釋性語言、批量化處理語言,大大的節省了工作成本
shell腳本第一行必須以 #!開頭,它表示該腳本使用後面的解釋器解釋執行。
舉個慄子:
//script.sh 註:這是一個文本文件
#!/bin/bash echo "this is a test" ls ls -l echo "there are all files"
執行方式:
//第一種執行方式: [admin@localhost Shell]$ chmod +x script.sh [admin@localhost Shell]$ ./script.sh //第二種執行方式: [admin@localhost Shell]$ /bin/bash script.sh
執行過程:
Shell會fork一個子進程並調用exec執行./script.sh這個程式,exec系統調用應該把子進程的代碼段替換成./script.sh程式的代碼段,並從它的_start開始執行。然而script.sh是個文本文件,根 本沒有代碼段和_start函數,怎麼辦呢?其實exec還有另外一種機制,如果要執行的是一個文本文件,並且第一行 指定瞭解釋器,則用解釋器程式的代碼段替換當前進程,並且從解釋器 的_start開始執行,而這個文本文件被當作命令行參數傳給解釋器。因此,執行上述腳本相當於執行程式!
輸入命令執行shell腳本後:
- 互動式進程(父進程)創建一個子進程用於執行腳本,父進程等待子進程終止
- 子進程程式替換bash解釋器
- 讀取shell腳本的命令,將其以參數傳遞的方式傳遞給bash解釋器
- 子bash對shell腳本傳入的參數進行讀取,讀一行識別到它是一個命令,則再創建一個子進程,子bash等待該新進程終止
- 新進程執行該命令,執行完後將結果交給子進程
- 子進程繼續讀取命令,創建新進程,新進程執行該命令,將結果返回給子進程,直到執行完最後一條命令
- 子進程終止,將結果返回給互動式父進程
註意:像export、cd、env、set 這些內置命令,在鍵入命令行後,互動式進程不會創建子進程,而是調用bash內部的函數執行這些命令,改變的是互動式進程。
如果在命令行下,將多個命令用括弧括起來,並用分號隔開來執行,互動式進程依然會創建一個子shell執行括弧中的命令:
如果不加括弧則是另一種情況,cd .. 命令會直接在互動式shell下執行:
. 或者 source這兩個命令是Shell的內建命令,這種方式不會創建子Shell,而是直接在互動式Shell下逐行 執行腳本中的命令。
例證:
script.sh
#!/bin/bash ls echo "#################" cd .. ls
shell變數
shell變數不需要進行任何聲明,直接定義即可,因為shell變數的值實際上都是字元串(對於沒有定義的變數預設是一個空串)。定義的時候shell變數由大寫字母加下劃線組成,並且定義的時候等號兩邊不能存在空格,否則會被認為是命令!
shell變數的種類:
- 環境變數:shell進程的環境變數可以從當前shell進程傳給fork出來的子進程。
- 本地變數:只存在於當前shell進程
利用printenv可以顯示當前shell進程的環境變數;利用set命令可以顯示當前shell進程中的定義的所有變數(包括環境變數和本地變數)和函數。
一個shell變數定義後僅存在於當前Shell進程,是一個本地變數。用export命令可以把本地變數導出為環境變數。用unset命令可以刪除已定義的環境變數或本地變數。
例如:
//分步 先定義後導出 COUNT=5 export COUNT //一步完成定義和導出環境變數 export COUNT=5 //刪除已經定義的環境變數 unset COUNT
變數引用:
引用shell變數要用到$符號,加{}可以防止歧義。
例如:
COUNT=5 echo $COUNT echo ${COUNT}911
通配符 Globbing、命令代換、單引號、雙引號
1.常見的通配符:
* : 匹配0個或多個任意字元
? : 匹配一個任意字元
[若幹字元] : 匹配方括弧中任意一個字元的一次出現
2.命令代換:
反引號`` 或者 $()
script.sh :
shell會先執行反引號或$()中的命令,將結果代換到當前命令行中!
算數代換:$(())
例如:將2-1的結果賦給SUB,並顯示本地變數SUB
SUB=$((2-1)) echo $SUB
3.單引號和雙引號
Shell腳本中的單引號和雙引號一樣都是字元串的界定符,單引號用於保持引號內所有字元的字面值,而雙引號有些情況則特殊。如果字元串中有特殊字元需要處理時就用雙引號。
註:單引號標註的字元中不能再出現單引號。
舉個慄子:
#!/bin/bash echo '\\' echo "\\" echo '`date`' echo "`date`"