bash腳本進階

過程式編程語言:
順序執行
選擇執行
if判斷
if 判斷條件;那么
條件為真的分支代碼
fi

if 判斷條件;then
條件為真的分支代碼
else
條件為假的分支代碼
fi
多分支
if 判斷條件1; then
條件為真的分支代碼
elif 判斷條件2; then
條件為真的分支代碼
elif 判斷條件3; then
條件為真的分支代碼
else
以上條件都為假的分支代碼
fi
case 變量引用 in
PAT1)
分支1
;;
PAT2)
分支2
;;

*)
默認分支
;;
esac
case支持glob風格的通配符:
*: 任意長度任意字符
?: 任意單個字符
[ ]:指定范圍內的任意單個字符
a|b: a或b

循環執行
循環:for,while,until
循環體:要執行的代碼;可能要執行n遍;
進入條件
退出條件
for循環:
for 變量名 in 列表;do
循環體
done
執行機制:
依次將列表中的元素賦值給“變量名”;每次賦值后即執行一次循環體,直到列表中的元素耗盡,循環結束
列表生成方式:
(1) 直接給出列表
(2) 整數列表:
(a) {start..end}
(b) $(seq [start [step]] end)
(3) 返回列表的命令
$(COMMAND)
(4) 使用glob,如:*.sh
使用通配符時用“”系統會認為是一個整體
(5) 變量引用;
$@, $*
腳本中使用多行重定向時結束語前加-解決縮進格式問題
例如:
for 條件;do
cat > a.txt <<-EOF
aaa
bbb
EOF
done
自加i++的返回值為1,++i的返回值為0
n=10; eval echo {1..$n} eval為掃描變量并替換
while循環
while 條件;do
循環體
done
循環控制條件:進入循環之前,先做一次判斷;每一次循環之后會再次做判斷;條件為“true”,則 執行一次循環;直到條件測試狀態為“false”終止循環
因此:循環條件一般應該有循環控制變量;而此變量的值會在循環體不斷地被修正
進入條件:循環條件為true
退出條件:循環條件為false
while死循環
while :;do
循環體
done
:相當于true,表示條件為真
until循環
until 循環條件;do
循環體
done
進入條件:循環條件為真
退出條件:循環條件為假
和while循環判斷條件正好相反
循環控制語句continue
用于循環體中
continue [N]:提前結束第N層的本輪循環,而直接進入下一輪判斷;最內層為第1層
while CONDTIITON1; do
CMD1

if CONDITION2; then
continue
fi
CMDn

done
循環控制語句break
用于循環體中
break [N]:提前結束第N層循環,最內層為第1層
while CONDTIITON1; do
CMD1

if CONDITION2; then
break
fi
CMDn

done
循環控制shift命令
shift [n]
用于將參量列表 list 左移指定次數,缺省為左移一次。
參量列表 list 一旦被移動,最左端的那個參數就從列表中刪除。while 循環遍歷位置參量列表時,常用到 shift
./doit.sh a b c d e f g h
./shfit.sh a b c d e f g h
創建無限循環
while true; do
循環體
done
until false; do
循環體
done
特殊用法
while循環的特殊用法(遍歷文件的每一行):
while read line; do
循環體
done < /PATH/FROM/SOMEFILE
依次讀取/PATH/FROM/SOMEFILE文件中的每一行,且將行賦值給變量line
雙小括號方法,即((…))格式,也可以用于算術運算
雙小括號方法也可以使bash Shell實現C語言風格的變量操作
I=10
((I++))
for循環的特殊格式:
for ((控制變量初始化;條件判斷表達式;控制變量的修正表達式))
do
循環體
done
控制變量初始化:僅在運行到循環代碼段時執行一次
控制變量的修正表達式:每輪循環結束會先進行控制變量修正運算,而后再做條件判斷
select循環與菜單
select variable in list
do
循環體命令
done
select 循環主要用于創建菜單,按數字順序排列的菜單項將顯示在
標準錯誤上,并顯示 PS3 提示符,等待用戶輸入
用戶輸入菜單列表中的某個數字,執行相應的命令
用戶輸入被保存在內置變量 REPLY 中
PS1:命令提示符;PS2:重定向提示符;PS3:select菜單提示符;$REPLY菜單中輸入的內容
select與case
select 是個無限循環,因此要記住用 break 命令退出循環,或用exit 命令終止腳本。也可以按 ctrl+c 退出循環
select 經常和 case 聯合使用
與 for 循環類似,可以省略 in list,此時使用位置參量
信號捕捉trap
trap ‘觸發指令’ 信號
自定義進程收到系統發出的指定信號后,將執行觸發指令,而不會執行原操作
trap ” 信號
忽略信號的操作
trap ‘-‘ 信號
恢復原信號的操作
trap -p
列出自定義信號操作
函數
函數function是由若干條shell命令組成的語句塊,實現代碼重用和模塊化編程
它與shell程序形式上是相似的,不同的是它不是一個單獨的進程,不能獨立運行,而是shell程序的一部分
函數和shell程序比較相似,區別在于:
Shell程序在子Shell中運行
而Shell函數在當前Shell中運行。因此在當前Shell中,函數可以對shell中變量進行修改
函數定義
函數由兩部分組成:函數名和函數體
語法一:
f_name (){
…函數體…
}
語法二:
function f_name {
…函數體…
}
語法三:
function f_name () {
…函數體…
}
declare -f [函數名] 查看系統函數庫或單個函數
{ 函數體 } 匿名函數
函數的優先級比所有的命令高
函數的生效范圍是當前shell
在函數中定義變量用 local ,只在函數內部生效,不影響全局變量
在函數中用declare -i 定義變量時自帶local屬性,不會影響全局變量
在函數中用declare -ig定義變量相當去普通變量,可以在函數外,當前shell中使用
action函數,實現顯示成功失敗功能,跟完整路徑 . /etc/init.d/functions

clipboard10

函數使用
函數的定義和使用:
可在交互式環境下定義函數
可將函數放在腳本文件中作為它的一部分
可放在只包含函數的單獨文件中
調用:函數只有被調用才會執行
調用:給定函數名
函數名出現的地方,會被自動替換為函數代碼
函數的生命周期:被調用時創建,返回時終止
函數返回值
函數有兩種返回值:
函數的執行結果返回值:
(1) 使用echo等命令進行輸出
(2) 函數體中調用命令的輸出結果
函數的退出狀態碼:
(1) 默認取決于函數中執行的最后一條命令的退出狀態碼
(2) 自定義退出狀態碼,其格式為:
return 從函數中返回,用最后狀態命令決定返回值
return 0 無錯誤返回。
return 1-255 有錯誤返回
交互式環境下定義和使用函數
示例:
dir() {
> ls -l
> }
定義該函數后,若在$后面鍵入dir,其顯示結果同ls -l的作用相同
dir
該dir函數將一直保留到用戶從系統退出,或執行了如下所示的unset命令
unset dir
unset可以同時刪除多個函數
在腳本中定義及使用函數
函數在使用前必須定義,因此應將函數定義放在腳本開始部分,直至shell首次發現它 后才能使用
調用函數僅使用其函數名即可
使用函數文件
可以將經常使用的函數存入函數文件,然后將函數文件載入shell
文件名可任意選取,但最好與相關任務有某種聯系。例如:functions.main
一旦函數文件載入shell,就可以在命令行或腳本中調用函數??梢允褂胹et命令查看所有定義的函數,其輸出列表包括已經載入shell的所有函數
若要改動函數,首先用unset命令從shell中刪除函數。改動完畢后,再重新載入此文件
載入函數
函數文件已創建好后,要將它載入shell
定位函數文件并載入shell的格式
. filename 或 source filename
注意:此即<點> <空格> <文件名>
這里的文件名要帶正確路徑
使用set命令檢查函數是否已載入。set命令將在shell中顯示所有的載入函數
要執行函數,簡單地鍵入函數名即可
刪除shell函數
現在對函數做一些改動后,需要先刪除函數,使其對shell不可用。使用unset命令完成刪除函數
命令格式為:
unset function_name
示例:
unset findit
再鍵入set命令,函數將不再顯示
環境函數
使子進程也可使用
聲明:export -f function_name
聲明函數不能直接寫在整個函數前,語法錯誤,只能單獨聲明成全局變量
查看:export -f 或 declare -xf
函數參數
函數可以接受參數:
傳遞參數給函數:調用函數時,在函數名后面以空白分隔給定參數列表即可;例如“testfunc arg1 arg2 …”
在函數體中當中,可使用$1, $2, …調用這些參數;還可以使用$@, $*, $#等特殊變量
函數變量
變量作用域:
環境變量:當前shell和子shell有效
本地變量:只在當前shell進程有效,為執行腳本會啟動專用子shell進程;因此,本地變量的作用范圍是當前shell腳本程序文件,包括腳本中的函數
局部變量:函數的生命周期;函數結束時變量被自動銷毀
注意:如果函數中有局部變量,如果其名稱同本地變量,使 用局部變量
name=wang
普通 標準 局部 全局
export name
環境 全局
function func() {local name}
本地
在函數中定義局部變量的方法
local NAME=VALUE
函數遞歸示例
函數遞歸:
函數直接或間接調用自身
注意遞歸層數
遞歸實例:
階乘是基斯頓·卡曼于 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)!
數組

變量:
存儲單個元素的內存空間,多個變量相同類型的集合
數組:
存儲多個元素的連續的內存空間,相當于多個變量的集合
數組名和索引
索引(下標):
編號從0開始,屬于數值索引
注意:索引可支持使用自定義的格式,而不僅是數值格式,即為關聯索引,
bash4.0版本之后開始支持
bash的數組支持稀疏格式(索引不連續)
聲明數組:
declare -a ARRAY_NAME(直接使用不用聲明)
declare -A ARRAY_NAME: 關聯數組,下標可以自定義(必須先聲明后使用)
例如:name[ceo]=laozi
注意:兩者不可相互轉換
數組賦值
數組元素的賦值
(1) 一次只賦值一個元素
數組名[下標]=值
weekdays[0]=”Sunday”
weekdays[4]=”Thursday”
(2) 一次賦值全部元素
數組名=(“VAL1” “VAL2” “VAL3″ …)
(3) 只賦值特定元素
數組名=([0]=”VAL1″ [3]=”VAL2″ …)
(4) 交互式數組值對賦值
read -a ARRAY
顯示所有數組:declare -a
引用數組
引用數組元素:
${數組名[INDEX]}
注意:省略[INDEX]表示引用下標為0的元素
引用數組所有元素:
${數組名[*]}
${數組名[@]}
數組的長度(數組中元素的個數):
${#數組名[*]}
${#數組名[@]}
刪除數組中的某元素:導致稀疏格式
unset數組名[下標]
刪除整個數組:
unset 數組名
數組數據處理
引用數組中的元素:
數組切片:
${ARRAY[@]:offset:number}
offset: 要跳過的元素個數
number: 要取出的元素個數
取偏移量之后的所有元素
${ARRAY[@]:offset}
向數組中追加元素:
ARRAY[${#ARRAY[*]}]=value
關聯數組:
declare -A ARRAY_NAME
ARRAY_NAME=([idx_name1]=’val1′ [idx_name2]=’val2‘…)
注意:關聯數組必須先聲明再調用
字符串切片
${#var}:返回字符串變量var的長度
${var:數字}:返回字符串變量var中從第幾個字符后(不包括第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個字符之間的內容
有版本要求,centos6上不能執行
注意:-length前空格
字符串處理
基于模式取子串
${var#*word}:其中word可以是指定的任意字符
功能:自左而右,查找var變量所存儲的字符串中,第一次出現的word, 刪除字符串開頭至第一次出現word字符之間的所有字符
${var##*word}:同上,貪婪模式,不同的是,刪除的是字符串開頭至最后一次由word指定的 字符之間的所有內容
${var%word*}:其中word可以是指定的任意字符
功能:自右而左,查找var變量所存儲的字符串中,第一次出現的word, 刪除字符串最后一個字符向左至第一次出現word字符之間的所有字符
file=”/var/log/messages”
${file%/*}: /var/log
${var%%word*}:同上,只不過刪除字符串最右側的字符向左至最后一次出現word字符之間的所有字符;
查找替換
${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所表示的字符串中所有以pattern為行首所匹配到的字符串
${var/%pattern}:刪除var所表示的字符串中所有以pattern為行尾所匹配到的字符串
字符大小寫轉換
${var^^}:把var中的所有小寫字母轉換為大寫
${var,,}:把var中的所有大寫字母轉換為小寫

clipboard11

高級變量用法-有類型變量
Shell變量一般是無類型的,但是bash Shell提供了declare和typeset兩個命令用于指定變量的類型,兩個命令是等價的
declare [選項] 變量名
-r 聲明或顯示只讀變量
-i 將變量定義為整型數
-a 將變量定義為數組
-A 將變量定義為關聯數組
-f 顯示已定義的所有函數名及其內容
-F 僅顯示已定義的所有函數名
-x 聲明或顯示環境變量和函數
-l 聲明變量為小寫字母 declare –l var=UPPER
-u 聲明變量為大寫字母 declare –u var=lower
eval命令
eval命令將會首先掃描命令行進行所有的置換,然后再執行該命令。該命令適用于那些一次掃描無法實現其功能的變量.該命令對變量進行兩次掃描
間接變量引用
如果第一個變量的值是第二個變量的名字,從第一個變量引用第二個變量的值就稱為間接變量引用
variable1的值是variable2,而variable2又是變量名,variable2的值為value,間接變量引用是指通過variable1獲得變量值value的行為
variable1=variable2
variable2=value
bash Shell提供了兩種格式實現間接變量引用
eval tempvar=\$$variable1
tempvar=${!variable1}
創建臨時文件
mktemp命令:創建并顯示臨時文件,可避免沖突
mktemp [OPTION]… [TEMPLATE]
TEMPLATE: filenameXXX
X至少要出現三個
OPTION:
-d: 創建臨時目錄
-p DIR或–tmpdir=DIR:指明臨時文件所存放目錄位置
安裝復制文件
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
expect介紹
expect 是由Don Libes基于Tcl( Tool Command Language )語言開發的,主要應用于自動化交互式操作的場景,借助Expect處理交互的命令,可以將交互過程如:ssh登錄,ftp登錄等寫在一個腳本上,使之自動化完成。尤其適用于需要對多臺服務器執行相同操作的環境中,可以大大提高系統管理人員的工作效率
expect命令
expect 語法:
expect [選項] [ -c cmds ] [ [ -[f|b] ] cmdfile ] [ args ]
選項
-c:從命令行執行expect腳本,默認expect是交互地執行的
示例:expect -c ‘expect “\n” {send “pressed enter\n”}
-d:可以輸出輸出調試信息
示例:expect -d ssh.exp
expect中相關命令
spawn:啟動新的進程
send:用于向進程發送字符串
expect:從進程接收字符串
interact:允許用戶交互
exp_continue 匹配多個字符串在執行動作后加此命令
expect最常用的語法(tcl語言:模式-動作)
單一分支模式語法:
expect “hi” {send “You said hi\n”}
匹配到hi后,會輸出“you said hi”,并換行
多分支模式語法:
expect “hi” { send “You said hi\n” } \
“hehe” { send “Hehe yourself\n” } \
“bye” { send “Good bye\n” }
匹配hi,hello,bye任意字符串時,執行相應輸出。等同如下:
expect {
“hi” { send “You said hi\n”}
“hehe” { send “Hehe yourself\n”}
“bye” { send “Good bye\n”}
}

本文來自投稿,不代表Linux運維部落立場,如若轉載,請注明出處:http://www.www58058.com/98167

(0)
老司機丨快上車老司機丨快上車
上一篇 2018-05-11
下一篇 2018-05-11

相關推薦

  • Linux的哲學思想

    1.一切皆文件
    2.小型,單一用途的程序
    3.配置信息保存在文本文件中
    4.盡量避免與用戶交互

    Linux筆記 2018-05-11
  • liunx 命令幫助

    — help和-h ? 選項如下:

    Linux筆記 2018-04-01
  • shell編程基礎二

    profile bashrc 混合命令行展開

    2018-04-11
  • 課堂實驗

    實驗:針對不同的用戶設置不同的共享目錄和權限 vim /etc/samba/smb.conf[global]workgroup = WANGGROUPsecurity = usernetbios name = smbsrv7passdb backend = tdbsamlog file = /var/log/samba/log.%Ilog level = 2…

    2018-07-01
  • 文本三劍客之grep及正則表達式用法

    grep (縮寫來自Globally search a Regular Expression and Print)是一種強大的文本搜索工具,與sed(文本過濾工具),awk(文本報告生成器)并稱為Linux文本處理三劍客,它能使用特定模式匹配(包括正則表達式)搜索文本,并默認輸出匹配行。

    2018-04-07
  • vim

    vim

    Linux筆記 2018-05-08
欧美性久久久久