函數介紹:
函數function 是由若干條shell 命令組成的語句塊,實現代碼重用和模塊化編程。
它與shell 程序形式上是相似的,不同的是它不是一個單獨的進程,不能獨立運行,而是shell 程序的一部分。
函數和shell 程序比較相似,區別在于:
Shell 程序在子Shell 中運行
而Shell 函數在當前Shell 中運行。因此在當前Shell 中,函數可以
對shell 中變量進行修改
函數由兩部分組成:函數名和函數體。
語法一:
function f_name {
… 函數體…
}
語法二:
function f_name ) () {
… 函數體…
}
語法三:
f_name (){
… 函數體…
}
函數的定義和使用:
? 可在交互式環境下定義函數
? 可將函數放在腳本文件中作為它的一部分
? 可放在只包含函數的單獨文件中
調用:函數只有被調用才會執行
調用:給定函數名函數名出現的地方,會被自動替換為函數代碼
函數的生命周期:被調用時創建,返回時終止
函數有兩種返回值:
函數的執行結果返回值:
(1) 使用echo 等命令進行輸出
(2) 函數體中調用命令的輸出結果
函數的退出狀態碼:
(1) 默認取決于函數中執行的最后一條命令的退出狀態碼
(2) 自定義退出狀態碼, 其格式為:
return 從函數中返回,用最后狀態命令決定返回值
return 0 無錯誤返回。
return 1-255 有錯誤
函數在使用前必須定義,因此應將函數定義放在腳本開始部分,直至shell 首次發現它后才能使用
可以將經常使用的函數存入函數文件,然后將函數文件載入shell
文件名可任意選取,但最好與相關任務有某種聯系。例如
:functions.main
一旦函數文件載入shell ,就可以在命令行或腳本中調用函數。可以使用set 命令查看所有定義的函數,其輸出列表包括已經載入shell 的所有函數
若要改動函數,首先用unset命令從shell 中刪除函數。改動完畢后,再重新載入此文件
函數可以接受參數:
傳遞參數給函數:調用函數時,在函數名后面以空白分隔
給定參數列表即可;例如“testfunc arg1 arg2 …”
在函數體中當中,可使用$1, $2, … 調用這些參數;還
可以使用$@, $*, $# 等特殊變量
變量作用域:
環境變量:當前shell 和子shell 有效
本地變量:只在當前shell 進程有效,為執行腳本會啟動
專用子shell 進程;因此,本地變量的作用范圍是當前shell 腳本
程序文件,包括腳本中的函數
局部變量:函數的生命周期;函數結束時變量被自動銷毀
注意:如果函數中有局部變量,如果其名稱同本地變量,使用局部變量。
在函數中定義局部變量的方法
local NAME=VALUE
環境函數:
exprot -f functions #必須先聲明環境函數,再定義
查看環境函數:export -f
函數遞歸:
函數直接或間接調用自身
注意遞歸層數
遞歸實例:
階乘是基斯頓·卡曼 于1808 年發明的運算符號,是數學術語
一個正整數的階乘(factorial )是所有小于及等于該數的正整
數的積,并且有0 的階乘為 1,自然數n 的階乘寫作n!
n!=1 ×2 ×3 ×… ×n
階乘亦可以遞歸方式定義:0!=1 ,n!=(n-1)! ×n
n!=n(n-1)(n-2)…1
n(n-1)! = n(n-1)(n-2)!
示例: fact.sh
!/bin/bash
fact() {
if [ $1 -eq 0 -o $1 -eq 1 ]; then
echo 1
else
echo $[$1$(fact $[$1-1])]
fi
}
fact $1
fork炸彈:
:(){ :|:& };:
bomb(){ bomb|bomb & };bomb #原理:遞歸調用,開啟無窮子shell占用內存空間
數組:
變量:存儲單個元素的內存空間
數組:存儲多個元素的連續的內存空間,相當于多個變量的
集合
數組名和索引
索引:編號從0 開始,屬于數值索引
注意:索引可支持使用自定義的格式,而不僅是數值格式
,即為關聯索引,bash4.0 版本之后開始支持
bash 的數組支持稀疏格式(索引不連續)
聲明數組:
declare -a ARRAY_NAME
declare -A ARRAY_NAME: 關聯數組 #關聯數組必須先聲明才能使用
而且數組和關聯數組不能互相轉換
unset shuzu : 刪除數組
數組賦值:
數組 元素的賦值:
(1) 一次只賦值一個元素;
ARRAY_NAME[INDEX]=VALUE
weekdays[0]=”Sunday”
weekdays[4]=”Thursday”
(2) 一次賦值全部元素:
ARRAY_NAME=(“VAL1” “VAL2” “VAL3” …)
(3) 只賦值特定元素:
ARRAY_NAME=([0]=”VAL1″ [3]=”VAL2″ …)
(4) 交互式數組值對賦值
read -a ARRAY
0 1 2 3 :每個數組值空格隔開
引用數組元素:${ARRAY_NAME[INDEX]}
注意:省略[INDEX] 表示引用下標為0 的元素
數組的長度( 數組中元素的個數):
${#ARRAY_NAME[]}
${#ARRAY_NAME[@]}
file=(/var/log/*.log) #支持通配符
arr=({1..10}) #1到10的數組
arr=({a..z})
arr=({a.b.c}.{txt,log}) #交互組合,生成六個數組
引用數組中的元素:
所有元素:${ARRAY[@]}, ${ARRAY[]}
數組切片:${ARRAY[@]:offset:number}
offset: 要跳過的元素個數
number: 要取出的元素個數
取偏移量之后的所有元素
${ARRAY[@]:offset}
examp: echo ${alpha[]:3:6} #去掉前三個數值,取后面6個;
echo ${alpha[]: -3} #從后往前取,一定要空格,從后往前取三個
echo ${alpha[]:3:-10} #去掉前面3個和后面10個取中間 負數再后面可以不空格
echo ${alpha[]: -4:-1} #從后取四個,再從后去掉一個,就只剩下三個了
${#var}: 返回字符串變量var 的長度
${var:offset}: 返回 字符串變量var 中從第offset 個字符后(不
包括第offset 個字符)的字符開始,到最后的部分,offset的 的
取值在0 到 到 ${#var}-1 之間(bash4.2 后,允許為負值)
${var:offset:number} :返回 字符串變量var 中從第offset個 個
字符 后(不包括第offset 個字符)的字符開始 ,長度為
number 的部分
${var: -length} :取字符串的最右側幾個字符
注意:冒號后必須有一空白字符
${var:offset:-length} :從最左側跳過offset 字符,一直向右
取到距離最右側lengh 個字符之前的內容
${var: -length:-offset} :先從最右側向左取到length 個字
符開始,再向右取到距離最右側offset
基于模式取子串
${var#word} :其中word 可以是指定的任意字符
功能:自左而右,查找var 變量所存儲的字符串中,第一
次出現的word, 刪除字符串開頭至第一次出現word 字符之間的所有字符
${var##word} :同上,貪婪模式,不同的是,刪除的
示例:
file=”var/log/messages“
${file#/}: log/messages
${file##/}: messages
${var%word} :其中word 可以是指定的任意字符;
功能:自右而左,查找var 變量所存儲的字符串中,第一
次出現的word, 刪除字符串最后一個字符向左至第一次出現word 字符之間的所有字符;
file=”/var/log/messages”
${file%/}: /var/log
${var%%word} :同上,只不過刪除字符串最右側的字符向左至最后一次出現word
字符之間的所有字符;
示例:
url=http://www.magedu.com:80
${url##:} 80
${url%%:} http
號從前往后,%從后往前
向數組中追加元素:
ARRAY[${#ARRAY[*]}]=需要追加的變量值 # 因為數組數標最大為n-1,所以增加的數組為n
刪除數組中的某元素:導致稀疏格式
unset ARRAY[INDEX] #可以刪除數字 ,也可以只刪除數組中的某元素
關聯數組:
declare -A ARRAY_NAME 注意:必須先聲明,再使用
ARRAY_NAME=([idx_name1]=’val1′ [idx_name2]=’val2‘…)
查找替換
${var/pattern/substr} :查找var 所表示的字符串中,第
一次被pattern 所匹配到的字符串,以substr 替換之
${var//pattern/substr}: 查找var 所表示的字符串中,
所有能被pattern 所匹配到的字符串,以substr 替換之
${var/#pattern/substr} :查找var 所表示的字符串中,
行首被pattern 所匹配到的字符串,以substr 替換之
${var/%pattern/substr} :查找var 所表示的字符串中,
行尾被pattern 所匹配到的字符串,以substr
查找并刪除
${var/pattern} :查找var 所表示的字符串中,刪除第一次被pattern 所匹配到的字符串
${var//pattern} :所有
${var/#pattern} :行首
${var/%pattern} :行尾
字符大小寫轉換
${var^^} :把var 中的所有小寫字母轉換為大寫
${var,,} :把var 中的所有大寫字母轉換為小寫
變量賦值:
${var:-value} :如果var 為空或未設置,那么返回value ;否則返回var的值,可省略:
${var:+value} :如果var 不空,則返回value ,否則返回空值;
${var:=value}:如果var為空或未設置,那么返回value并將value賦值給var,否則返回var的值
${var:?error_info}:如果var 為空或未設置,那么在當前終端打印error_info ;否則返回var
Shell 變量一般是無類型的,但是bash Shell 提供了declare和
typeset 兩個命令用于指定變量的類型,兩個命令是等價的
declare [ 選項] 變量名
-r 聲明或顯示只讀變量
-i 將變量定義為整型數
-a 將變量定義為數組
-A 將變量定義為關聯數組
-f 顯示此腳本前定義過的所有函數名及其內容
-F 僅顯示此腳本前定義過的所有函數名
-x 聲明或顯示環境變量和函數
-l 聲明變量為小寫字母 declare –l var=UPPER
-u 聲明變量為大寫字母 declare –u var=lower
eval 命令將會首先掃描命令行進行所有的置換,然后再執行該
命令。該命令適用于那些一次掃描無法實現其功能的變量.該
命令對變量進行兩次掃描
間接變量引用:
如果第一個變量的值是第二個變量的名字,從第一個變量引
用第二個變量的值就稱為間接變量引用
variable1 的值是variable2 ,而variable2 又是變量名,
variable2 的值為value ,間接變量引用是指通過variable1獲得變量值value 的行為
variable1=variable2
variable2=value
創建臨時文件:
mktemp 命令:創建并顯示臨時文件,可避免沖突
mktemp [OPTION]… [TEMPLATE]
TEMPLATE: filename.XXX
X 至少要出現三個
OPTION:
-d: 創建臨時目錄
-p DIR 或–tmpdir=DIR
examp:mktemp –tmpdir=/testdir test.XXXXXX #直接選擇全路徑創建此選項可不用
install: 類似cp chmod chgrp mkdir 等集合。用法和cp命令差不多
install 命令:
install [OPTION]… [-T] SOURCE DEST 單文件
install [OPTION]… SOURCE… DIRECTORY
install [OPTION]… -t DIRECTORY SOURCE…
install [OPTION]… -d DIRECTORY… 創建空目錄
選項:
-m MODE ,默認755 #直接復制過去指定權限和屬主屬組
-o OWNER
-g GROUP
原創文章,作者:shewei,如若轉載,請注明出處:http://www.www58058.com/72246