1、
Shell腳本中用
#
表示注釋,相當(dāng)于C語(yǔ)言的
//
注釋。但如果
#
位于第一行開頭,并且是
#!
(稱為Shebang
)則例外,它表示該腳本使用?后面指定的解釋器
/bin/sh
解釋執(zhí)行
?
$ chmod +x script.sh $ ./script.sh
?
2、兩種執(zhí)行Shell腳本的方法:
$ ./script.sh $ sh ./script.sh
3、 一行中可以輸入由分號(hào);隔開的多個(gè)命令
?
$ cd ..;ls -l
4、
只存在于當(dāng)前Shell進(jìn)程,用
set
命令可以顯示當(dāng)前Shell進(jìn)程中定義的所有變量(包括本地變量和環(huán)境變量)和函數(shù)
?
?
環(huán)境變量是任何進(jìn)程都有的概念,而本地變量是Shell特有的概念。在Shell中,環(huán)境變量和本地變量的定義和用法相似。在Shell中定義或賦值一個(gè)變量:
$ VARNAME=value
注意等號(hào)兩邊都不能有空格,否則會(huì)被Shell解釋成命令和命令行參數(shù)。
一個(gè)變量定義后僅存在于當(dāng)前Shell進(jìn)程,它是本地變量,用
export
命令可以把本地變量導(dǎo)出為環(huán)境變量,定義和導(dǎo)出環(huán)境變量通常可以一步完成:
$ export VARNAME=value
也可以分兩步完成:
$ VARNAME=value $ export VARNAME
用
unset
命令可以刪除已定義的環(huán)境變量或本地變量。
$ unset VARNAME
如果一個(gè)變量叫做
VARNAME
,用
${VARNAME}
可以表示它的值,在不引起歧義的情況下也可以用
$VARNAME
表示它的值。通過(guò)以下例子比較這兩種表示法的不同:
$ echo $SHELL $ echo $SHELLabc $ echo $SHELL abc $ echo ${SHELL}abc
注意,在定義變量時(shí)不用$,取變量值時(shí)要用$。和C語(yǔ)言不同的是,Shell變量不需要明確定義類型,事實(shí)上Shell變量的值都是字符串,比如我們定義
VAR=45
,其實(shí)
VAR
的值是字符串
45
而非整數(shù)。Shell變量不需要先定義后使用,如果對(duì)一個(gè)沒有定義的變量取值,則值為空字符串。
5、Shell腳本中的單引號(hào)和雙引號(hào)一樣都是字符串的界定符(雙引號(hào)下一節(jié)介紹),而不是字符的界定符。單引號(hào)用于保持引號(hào)內(nèi)所有字符的字面值,即使引號(hào)內(nèi)的\和回車也不例外,但是字符串中不能出現(xiàn)單引號(hào)。如果引號(hào)沒有配對(duì)就輸入回車,Shell會(huì)給出續(xù)行提示符,要求用戶把引號(hào)配上對(duì)。例如:
$ echo '$SHELL' $SHELL $ echo 'ABC\(回車) > DE'(再按一次回車結(jié)束命令) ABC\ DE
6、 命令代換:`或 $()
由反引號(hào)括起來(lái)的也是一條命令,Shell先執(zhí)行該命令,然后將輸出結(jié)果立刻代換到當(dāng)前命令行中。例如定義一個(gè)變量存放
date
命令的輸出:
$ DATE=`date` $ echo $DATE
命令代換也可以用
$()
表示:
$ DATE=$(date)
7、
雙引號(hào)用于保持引號(hào)內(nèi)所有字符的字面值(回車也不例外),但以下情況除外:
-
$加變量名可以取變量的值
-
反引號(hào)仍表示命令替換
-
\$表示$的字面值
-
\`表示`的字面值
-
\"表示"的字面值
-
\\表示\的字面值
-
除以上情況之外,在其它字符前面的\無(wú)特殊含義,只表示字面值
$ echo "$SHELL" /bin/bash $ echo "`date`" Sun Apr 20 11:22:06 CEST 2003 $ echo "I'd say: \"Go for it\"" I'd say: "Go for it" $ echo "\"(回車) >"(再按一次回車結(jié)束命令) " $ echo "\\" \
8、
啟動(dòng)
bash
會(huì)自動(dòng)執(zhí)行以下腳本:
-
首先執(zhí)行
/etc/profile
,系統(tǒng)中每個(gè)用戶登錄時(shí)都要執(zhí)行這個(gè)腳本,如果系統(tǒng)管理員希望某個(gè)設(shè)置對(duì)所有用戶都生效,可以寫在這個(gè)腳本里 -
然后依次查找當(dāng)前用戶主目錄的
~/.bash_profile
、~/.bash_login
和~/.profile
三個(gè)文件,找到第一個(gè)存在并且可讀的文件來(lái)執(zhí)行,如果希望某個(gè)設(shè)置只對(duì)當(dāng)前用戶生效,可以寫在這個(gè)腳本里,由于這個(gè)腳本在/etc/profile
之后執(zhí)行,/etc/profile
設(shè)置的一些環(huán)境變量的值在這個(gè)腳本中可以修改,也就是說(shuō),當(dāng)前用戶的設(shè)置可以覆蓋(Override) 系統(tǒng)中全局的設(shè)置。~/.profile
這個(gè)啟動(dòng)腳本是sh
規(guī)定的,bash
規(guī)定首先查找以~/.bash_
開頭的啟動(dòng)腳本,如果沒有則執(zhí)行~/.profile
,是為了和sh
保持一致。 -
順便一提,在退出登錄時(shí)會(huì)執(zhí)行
~/.bash_logout
腳本(如果它存在的話)。
?
9、條件測(cè)試test
?
命令
test
或
[
可以測(cè)試一個(gè)條件是否成立,如果測(cè)試結(jié)果為真,則該命令的Exit Status為0,如果測(cè)試結(jié)果為假,則命令的Exit Status為1(注意與C語(yǔ)言的邏輯表示正好相反)。例如測(cè)試兩個(gè)數(shù)的大小關(guān)系:
$ VAR=2 $ test $VAR -gt 1 $ echo $? 0 $ test $VAR -gt 3 $ echo $? 1 $ [ $VAR -gt 3 ] $ echo $? 1
雖然看起來(lái)很奇怪,但左方括號(hào)
[
確實(shí)是一個(gè)命令的名字,傳給命令的各參數(shù)之間應(yīng)該用空格隔開
,比如,
$VAR
、
-gt
、
3
、
]
是
[
命令的四個(gè)參數(shù),它們之間必須用空格隔開。命令
test
或
[
的參數(shù)形式是相同的,只不過(guò)
test
命令不需要
]
參數(shù)。以
[
命令為例,常見的測(cè)試命令如下表所示:
[ -d DIR ]
|
如果
DIR
存在并且是一個(gè)目錄則為真
|
[ -f FILE ]
|
如果
FILE
存在且是一個(gè)普通文件則為真
|
[ -z STRING ]
|
如果
STRING
的長(zhǎng)度為零則為真
|
[ -n STRING ]
|
如果
STRING
的長(zhǎng)度非零則為真
|
[ STRING1 = STRING2 ]
|
如果兩個(gè)字符串相同則為真 |
[ STRING1 != STRING2 ]
|
如果字符串不相同則為真 |
[ ARG1 OP ARG2 ]
|
ARG1
和
ARG2
應(yīng)該是整數(shù)或者取值為整數(shù)的變量,
OP
是
-eq
(等于)
-ne
(不等于)
-lt
(小于)
-le
(小于等于)
-gt
(大于)
-ge
(大于等于)之中的一個(gè)
|
?
和C語(yǔ)言類似,測(cè)試條件之間還可以做與、或、非邏輯運(yùn)算:
帶與、或、非的測(cè)試命令
[ ! EXPR ]
|
EXPR
可以是上表中的任意一種測(cè)試條件,!表示邏輯反
|
[ EXPR1 -a EXPR2 ]
|
EXPR1
和
EXPR2
可以是上表中的任意一種測(cè)試條件,
-a
表示邏輯與
|
[ EXPR1 -o EXPR2 ]
|
EXPR1
和
EXPR2
可以是上表中的任意一種測(cè)試條件,
-o
表示邏輯或
|
?
例如:
$ VAR=abc $ [ -d Desktop -a $VAR = 'abc' ] $ echo $? 0
注意,如果上例中的
$VAR
變量事先沒有定義,則被Shell展開為空字符串,會(huì)造成測(cè)試條件的語(yǔ)法錯(cuò)誤(展開為
[ -d Desktop -a = 'abc' ]
),作為一種好的Shell編程習(xí)慣,應(yīng)該總是把變量取值放在雙引號(hào)之中(展開為
[ -d Desktop -a "" = 'abc' ]
):
$ unset VAR $ [ -d Desktop -a $VAR = 'abc' ] bash: [: too many arguments $ [ -d Desktop -a "$VAR" = 'abc' ] $ echo $? 1
10、 case/esac
case
命令可類比C語(yǔ)言的
switch
/
case
語(yǔ)句,
esac
表示
case
語(yǔ)句塊的結(jié)束。C語(yǔ)言的
case
只能匹配整型或字符型常量表達(dá)式,而Shell腳本的
case
可以匹配字符串和Wildcard,每個(gè)匹配分支可以有若干條命令,末尾必須以;;結(jié)束,執(zhí)行時(shí)找到第一個(gè)匹配的分支并執(zhí)行相應(yīng)的命令,然后直接跳到
esac
之后,不需要像C語(yǔ)言一樣用
break
跳出。
#! /bin/sh echo "Is it morning? Please answer yes or no." read YES_OR_NO case "$YES_OR_NO" in yes|y|Yes|YES) echo "Good Morning!";; [nN]*) echo "Good Afternoon!";; *) echo "Sorry, $YES_OR_NO not recognized. Enter yes or no." exit 1;; esac exit 0
使用
case
語(yǔ)句的例子可以在系統(tǒng)服務(wù)的腳本目錄
/etc/init.d
中找到。這個(gè)目錄下的腳本大多具有這種形式(以
/etc/apache2
為例):
case $1 in start) ... ;; stop) ... ;; reload | force-reload) ... ;; restart) ... *) log_success_msg "Usage: /etc/init.d/apache2 {start|stop|restart|reload|force-reload|start-htcacheclean|stop-htcacheclean}" exit 1 ;; esac
啟動(dòng)
apache2
服務(wù)的命令是
$ sudo /etc/init.d/apache2 start
$1
是一個(gè)特殊變量,在執(zhí)行腳本時(shí)自動(dòng)取值為第一個(gè)命令行參數(shù),也就是
start
,所以進(jìn)入
start)
分支執(zhí)行相關(guān)的命令。同理,命令行參數(shù)指定為
stop
、
reload
或
restart
可以進(jìn)入其它分支執(zhí)行停止服務(wù)、重新加載配置文件或重新啟動(dòng)服務(wù)的相關(guān)命令。
11、if fi
和C語(yǔ)言類似,在Shell中用
if
、
then
、
elif
、
else
、
fi
這幾條命令實(shí)現(xiàn)分支控制。這種流程控制語(yǔ)句本質(zhì)上也是由若干條Shell命令組成的,例如先前講過(guò)的
if [ -f ~/.bashrc ]; then . ~/.bashrc fi
其實(shí)是三條命令,
if [ -f ~/.bashrc ]
是第一條,
then . ~/.bashrc
是第二條,
fi
是第三條。如果兩條命令寫在同一行則需要用;號(hào)隔開,一行只寫一條命令就不需要寫;號(hào)了,另外,
then
后面有換行,但這條命令沒寫完,Shell會(huì)自動(dòng)續(xù)行,把下一行接在
then
后面當(dāng)作一條命令處理。和
[
命令一樣,要注意命令和各參數(shù)之間必須用空格隔開。
if
命令的參數(shù)組成一條子命令,如果該子命令的Exit Status為0(表示真),則執(zhí)行
then
后面的子命令,如果Exit Status非0(表示假),則執(zhí)行
elif
、
else
或者
fi
后面的子命令。
if
后面的子命令通常是測(cè)試命令,但也可以是其它命令。Shell腳本沒有{}括號(hào),所以用
fi
表示
if
語(yǔ)句塊的結(jié)束。見下例:
#! /bin/sh if [ -f /bin/bash ] then echo "/bin/bash is a file" else echo "/bin/bash is NOT a file" fi if :; then echo "always true"; fi
:
是一個(gè)特殊的命令,稱為空命令,該命令不做任何事,但Exit Status總是真。此外,也可以執(zhí)行
/bin/true
或
/bin/false
得到真或假的Exit Status。再看一個(gè)例子:
#! /bin/sh echo "Is it morning? Please answer yes or no." read YES_OR_NO if [ "$YES_OR_NO" = "yes" ]; then echo "Good morning!" elif [ "$YES_OR_NO" = "no" ]; then echo "Good afternoon!" else echo "Sorry, $YES_OR_NO not recognized. Enter yes or no." exit 1 fi exit 0
上例中的
read
命令的作用是等待用戶輸入一行字符串,將該字符串存到一個(gè)Shell變量中。
此外,Shell還提供了&&和||語(yǔ)法,和C語(yǔ)言類似,具有Short-circuit特性,很多Shell腳本喜歡寫成這樣:
test "$(whoami)" != 'root' && (echo you are using a non-privileged account; exit 1)
&&相當(dāng)于“
if...then...
”,而||相當(dāng)于“
if not...then...
”。&&和||用于連接兩個(gè)命令,而上面講的
-a
和
-o
僅用于在測(cè)試表達(dá)式中連接兩個(gè)測(cè)試條件,要注意它們的區(qū)別,例如,
test "$VAR" -gt 1 -a "$VAR" -lt 3
和以下寫法是等價(jià)的
test "$VAR" -gt 1 && test "$VAR" -lt 3
更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號(hào)聯(lián)系: 360901061
您的支持是博主寫作最大的動(dòng)力,如果您喜歡我的文章,感覺我的文章對(duì)您有幫助,請(qǐng)用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長(zhǎng)非常感激您!手機(jī)微信長(zhǎng)按不能支付解決辦法:請(qǐng)將微信支付二維碼保存到相冊(cè),切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對(duì)您有幫助就好】元
