shell腳本中的函數(shù)與子shell
函數(shù)
函數(shù)的使用
bash中也有函數(shù)。一個(gè)函數(shù)就是一個(gè)子程序,是用于實(shí)現(xiàn)一串操作的代碼塊。bash中的函數(shù)的形式如下:
function function_name(){
}
function_name(){
}
function_name()
{
}
以上三種形式的函數(shù)均可達(dá)到目的。
使用函數(shù)的時(shí)候只要簡單的使用函數(shù)的名字即可。
例如:
#!/bin/bash
function abc(){
echo abc
}
abc
其輸出結(jié)果為:
abc
bash中的函數(shù)必須在其第一次調(diào)用之前被完成。沒有如c語言中那種函數(shù)聲明的形式。即使提前定義了函數(shù)的變量也不可以。但是只要執(zhí)行順序中,函數(shù)的定義在函數(shù)的使用之前即可。
考慮如下情況:
function f1(){
echo abc
f2
}
function f2(){
echo cde
}
f1
這種情況中,雖然看起來f2的使用在其定義之前已經(jīng)有過一次了,但是實(shí)際運(yùn)行當(dāng)中我們不難發(fā)現(xiàn)f2先定義了之后f1才調(diào)用了f2。因此這種情況也是可以的。
函數(shù)是可以嵌套的。
考慮如下情況:
f1(){
f2(){
echo abc
}
}
f2
f1
f2
其輸出結(jié)果類似于:
bash: f2: command not found
abc
第一行表示沒有找到f2,第二行執(zhí)行了f1,第三行找到了f2。
函數(shù)聲明也可能出現(xiàn)在任何看起來不可能的應(yīng)該有一個(gè)命令出現(xiàn)的地方。
由以上內(nèi)容可以看出,其實(shí)函數(shù)的定義其實(shí)只是一種命令。而bash腳本是一種線性執(zhí)行的語言。因此只要保證函數(shù)使用的流程正確,就可以將函數(shù)應(yīng)用在各種可能的地方。
函數(shù)的傳參
bash的函數(shù)能夠接受參數(shù)并返回狀態(tài)退出碼。
函數(shù)以位置來引用傳遞來的參數(shù),類似腳本的參數(shù)傳遞。例如2等等。
例如:
function abc(){
echo $1
return 22
}
abc cde
echo $?
以上內(nèi)容為一個(gè)最簡單的函數(shù)的傳參及返回值的情況。
輸出為:
cde
22
此外,shift命令也可以應(yīng)用于傳遞給函數(shù)的參數(shù),像工作在腳本的參數(shù)中那樣。
函數(shù)接受參數(shù)通常只是接受參數(shù)的值。因此我們也可以傳遞間接引用給函數(shù)。
例如:
function echoabc(){
echo $1
}
a=b
b=c
echoabc ${a}
echoabc ${!a}
b=d
echoabc ${a}
echoabc ${!a}
以上內(nèi)容的輸出為:
b
c
b
d
與此同理,我們可以在函數(shù)中修改間接引用中的值:
#!/bin/bash
function changeabc(){
echo $1
eval "echo \$$1"
eval "$1=xyz"
eval "echo \$$1"
}
var1=var2
var2=abc
changeabc $var1
echo $var2
eval "echo \$$var1"
以上內(nèi)容的輸出為:
var2
abc
xyz
xyz
xyz
函數(shù)的退出碼
函數(shù)返回一個(gè)被稱為退出狀態(tài)的值. 退出狀態(tài)可以由return來指定statement, 否則函數(shù)的退出狀態(tài)是函數(shù)最后一個(gè)執(zhí)行命令的退出狀態(tài)(0表示成功,非0表示出錯(cuò)代碼). 退出狀態(tài)(exit status)可以在腳本中由$? 引用. 這個(gè)機(jī)制使腳本函數(shù)也可以像C函數(shù)一樣有一個(gè)"返回值".
可以用來返回異常狀態(tài)。
函數(shù)的重定向
函數(shù)的本質(zhì)也是一個(gè)代碼塊,因此我們可以重定向其標(biāo)準(zhǔn)輸入輸出。
例如:
function echoabc(){
read line
while [ -n "$line" ]
do
echo $line
read line
done
} << strings
abcd
cdef
strings
echoabc
將一個(gè)heredocuments重定向到函數(shù)echoabc的輸入。輸出為:
abcd
cdef
子shell
運(yùn)行一個(gè)shell腳本時(shí)會(huì)啟動(dòng)另一個(gè)命令解釋器. 就好像你的命令是在命令行提示下被解釋的一樣, 類似于批處理文件里的一系列命令.每個(gè)shell腳本有效地運(yùn)行在父shell(parent shell)的一個(gè)子進(jìn)程里.這個(gè)父shell是指在一個(gè)控制終端或在一個(gè)xterm窗口中給你命令指示符的進(jìn)程.
shell腳本也能啟動(dòng)他自已的子進(jìn)程. 這些子shell(即子進(jìn)程)使腳本并行地,有效率地地同時(shí)運(yùn)行多個(gè)子任務(wù).
圓括號運(yùn)行子shell
嵌在圓括號里的一列命令在一個(gè)子shell里運(yùn)行。
例如:
(
echo abcd
echo cdef
)
輸出為:
abcd
cdef
在子shell里的變量不能被這段子shell代碼塊之外外面的腳本訪問.這些變量是不能被產(chǎn)生這個(gè)子shell的父進(jìn)程(parent process)存取的,實(shí)際上它們是局部變量(local variables).
例如:
(a=b)
echo $a
輸出為空
在子shell中的目錄更改不會(huì)影響到父shell.
子shell可用于為一組命令設(shè)定臨時(shí)的環(huán)境變量
進(jìn)程在不同的子shell中可以并行地執(zhí)行.這樣就允許把一個(gè)復(fù)雜的任務(wù)分成幾個(gè)小的子問題來同時(shí)地處理。
管道產(chǎn)生子shell
管道(|)也會(huì)產(chǎn)生子shell。在子shell中可以讀取父shell中的變量,但是不能寫這些變量。有時(shí)我們可以通過重定向輸入輸出的方式來傳遞這些變量。