for 變量名 in 列表;do
?循環(huán)體
done成都網(wǎng)站設(shè)計(jì)、成都網(wǎng)站制作的關(guān)注點(diǎn)不是能為您做些什么網(wǎng)站,而是怎么做網(wǎng)站,有沒有做好網(wǎng)站,給成都創(chuàng)新互聯(lián)一個(gè)展示的機(jī)會(huì)來證明自己,這并不會(huì)花費(fèi)您太多時(shí)間,或許會(huì)給您帶來新的靈感和驚喜。面向用戶友好,注重用戶體驗(yàn),一切以用戶為中心。
依次將列表中的元素賦值給“變量名” ; 每次賦值后即執(zhí)行一次循環(huán)體; 直到列表中的元素耗盡,循環(huán)結(jié)束
雙小括號(hào)方法,即((…))格式,也可以用于算術(shù)運(yùn)算
雙小括號(hào)方法也可以使bash Shell實(shí)現(xiàn)C語言風(fēng)格的變量操作
I=10
((I++))
for ((控制變量初始化;條件判斷表達(dá)式;控制變量的修正表達(dá)式))
do
?循環(huán)體
done
- 控制變量初始化:僅在運(yùn)行到循環(huán)代碼段時(shí)執(zhí)行一次
- 控制變量的修正表達(dá)式:每輪循環(huán)結(jié)束會(huì)先進(jìn)行控制變量修正運(yùn)算,而后再做條件判斷
while CONDITION; do
?循環(huán)體
done
- CONDITION:循環(huán)控制條件;進(jìn)入循環(huán)之前,先做一次判斷;每一次循環(huán)之后會(huì)再次做判斷;條件為“true” ,則執(zhí)行一次循環(huán);直到條件測(cè)試狀態(tài)為“false”終止循環(huán)
- 因此:CONDTION一般應(yīng)該有循環(huán)控制變量;而此變量的值會(huì)在循環(huán)體不斷地被修正
- 進(jìn)入條件:CONDITION為true
- 退出條件:CONDITION為false
while read line; do
?循環(huán)體
done < /PATH/FROM/SOMEFILE
依次讀取/PATH/FROM/SOMEFILE文件中的每一行,且將行賦值給變量line
until CONDITION; do
?循環(huán)體
done
- 進(jìn)入條件: CONDITION 為false
- 退出條件: CONDITION 為true
- break
用于循環(huán)體中
提前結(jié)束循環(huán),最內(nèi)層為第一層- continue
用于循環(huán)體中
提前結(jié)束循環(huán),直接進(jìn)入下一輪判斷;最內(nèi)層位第1層- shift
用于將參量列表 list 左移指定次數(shù),缺省為左移一次
參量列表 list 一旦被移動(dòng),最左端的那個(gè)參數(shù)就從列表中刪除。 while 循環(huán)遍歷位置參量列表時(shí),常用到 shift
while true; do
?循環(huán)體
done
until false; do
?循環(huán)體
done
select variable in list
do
?循環(huán)體命令
done
select 循環(huán)主要用于創(chuàng)建菜單,按數(shù)字順序排列的菜單項(xiàng)將顯示在標(biāo)準(zhǔn)錯(cuò),并顯示 PS3 提示符,等待用戶輸入
用戶輸入菜單列表中的某個(gè)數(shù)字,執(zhí)行相應(yīng)的命令
用戶輸入被保存在內(nèi)置變量 REPLY 中
select 是個(gè)無限循環(huán),因此要記住用 break 命令退出循環(huán),或用 exit 命令終止腳本。也可以按 ctrl+c 退出循環(huán)
select 經(jīng)常和 case 聯(lián)合使用
與 for 循環(huán)類似,可以省略 in list,此時(shí)使用位置參量
函數(shù)function是由若干條shell命令組成的語句塊,實(shí)現(xiàn)代碼重用和模塊化編程
它與shell程序形式上是相似的,不同的是它不是一個(gè)單獨(dú)的進(jìn)程,不能獨(dú)立運(yùn)行,而是shell程序的一部分
函數(shù)和shell程序比較相似,區(qū)別在于
Shell程序在子Shell中運(yùn)行
而Shell函數(shù)在當(dāng)前Shell中運(yùn)行。因此在當(dāng)前Shell中,函數(shù)可以對(duì)shell中變量進(jìn)行修改
函數(shù)由兩部分組成:函數(shù)名和函數(shù)體
??f_name (){
???...函數(shù)體...
?}
??function f_name {
???...函數(shù)體...
?}
??function f_name () {
???...函數(shù)體...
?}函數(shù)使用
- 函數(shù)的定義和使用:
?可在交互式環(huán)境下定義函數(shù)
?可將函數(shù)放在腳本文件中作為它的一部分
?可放在只包含函數(shù)的單獨(dú)文件中- 調(diào)用:函數(shù)只有被調(diào)用才會(huì)執(zhí)行
?調(diào)用:給定函數(shù)名
?函數(shù)名出現(xiàn)的地方,會(huì)被自動(dòng)替換為函數(shù)代碼- 函數(shù)的生命周期:被調(diào)用時(shí)創(chuàng)建,返回時(shí)終止
函數(shù)有兩種返回值:
1、函數(shù)的執(zhí)行結(jié)果返回值:
?(1) 使用echo等命令進(jìn)行輸出
?(2) 函數(shù)體中調(diào)用命令的輸出結(jié)果
2、函數(shù)的退出狀態(tài)碼:
?(1) 默認(rèn)取決于函數(shù)中執(zhí)行的最后一條命令的退出狀態(tài)碼
?(2) 自定義退出狀態(tài)碼,其格式為:
?return 從函數(shù)中返回,用最后狀態(tài)命令決定返回值
?return 0 無錯(cuò)誤返回
?return 1-255 有錯(cuò)誤返回
函數(shù)在使用前必須定義,因此應(yīng)將函數(shù)定義放在腳本開始部分,直至shell首次發(fā)現(xiàn)它后才能使用
調(diào)用函數(shù)僅使用其函數(shù)名即可示例:
?cat func1
?#!/bin/bash
?# func1
?hello()
?{
???echo "Hello there today's date is date +%F"
?}
?echo "now going to the function hello"
?hello
?echo “back from the function”
- 可以將經(jīng)常使用的函數(shù)存入函數(shù)文件,然后將函數(shù)文件載入shell
- 文件名可任意選取,但最好與相關(guān)任務(wù)有某種聯(lián)系。例如:functions.main
- 一旦函數(shù)文件載入shell,就可以在命令行或腳本中調(diào)用函數(shù)??梢允褂胹et命令查看所有定義的函數(shù),其輸出列表包括已經(jīng)載入shell的所有函數(shù)
- 若要改動(dòng)函數(shù),首先用unset命令從shell中刪除函數(shù)。改動(dòng)完畢后,再重新載入此文件
函數(shù)文件示例:
?cat functions.main
?#!/bin/bash
?#functions.main
?findit()
?{
???if [ $# -lt 1 ] ; then
????echo "Usage:findit file"
????return 1
???fi
???find / -name $1 –print
?}
- 函數(shù)文件已創(chuàng)建好后,要將它載入shell
- 定位函數(shù)文件并載入shell的格式
?. filename 或 source filename- 注意:此即<點(diǎn)> <空格> <文件名>
- 這里的文件名要帶正確路徑
- 示例:
?上例中的函數(shù),可使用如下命令
?. functions.main
使用set命令檢查函數(shù)是否已載入。 set命令將在shell中顯示所有的載入函數(shù)
示例:
?set
?findit=( )
?{
???if [ $# -lt 1 ]; then
???echo "usage :findit file";
???return 1
???fi
???find / -name $1 -print
?}
?…
要執(zhí)行函數(shù),簡(jiǎn)單地鍵入函數(shù)名即可
示例:
?findit groups
?/usr/bin/groups
?/usr/local/backups/groups.bak
現(xiàn)在對(duì)函數(shù)做一些改動(dòng)后,需要先刪除函數(shù),使其對(duì)shell不可用。使用unset命令完成刪除函數(shù)
命令格式為:
?unset function_name示例:
?unset findit
?再鍵入set命令,函數(shù)將不再顯示
使子進(jìn)程也可使用
?聲明:export -f function_name
?查看:export -f 或 declare -xf
函數(shù)可以接受參數(shù):
?傳遞參數(shù)給函數(shù):調(diào)用函數(shù)時(shí),在函數(shù)名后面以空白分隔給定參數(shù)列表即可;例如“testfunc arg1 arg2 ...”
?在函數(shù)體中當(dāng)中,可使用$1, $2, ...調(diào)用這些參數(shù);還可以使用$@, $*, $#等特殊變量
- 變量作用域:
? 環(huán)境變量:當(dāng)前shell和子shell有效
? 本地變量:只在當(dāng)前shell進(jìn)程有效,為執(zhí)行腳本會(huì)啟動(dòng)專用子shell進(jìn)程;因此,本地變量的作用范圍是當(dāng)前shell腳本程序文件,包括腳本中的函數(shù)
? 局部變量:函數(shù)的生命周期;函數(shù)結(jié)束時(shí)變量被自動(dòng)銷毀- 注意:如果函數(shù)中有局部變量,如果其名稱同本地變量,使用局部變量
- 在函數(shù)中定義局部變量的方法
? local NAME=VALUE
函數(shù)直接或間接調(diào)用自身
注意遞歸層數(shù)示例: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進(jìn)程是一種惡意程序,它的內(nèi)部是一個(gè)不斷在fork進(jìn)程的無限循環(huán),實(shí)質(zhì)是一個(gè)簡(jiǎn)單的遞歸程序。由于程序是遞歸的,如果沒有任何限制,這會(huì)導(dǎo)致這個(gè)簡(jiǎn)單的程序迅速耗盡系統(tǒng)里面的所有資源
??:(){ :|:& };:
?bomb() { bomb | bomb & }; bomb
??cat Bomb.sh
?#!/bin/bash
?./$0|./$0&
trap '觸發(fā)指令' 信號(hào)
? 進(jìn)程收到系統(tǒng)發(fā)出的指定信號(hào)后,將執(zhí)行自定義指令,而不會(huì)執(zhí)行原操作
trap '' 信號(hào)
? 忽略信號(hào)的操作
trap '-' 信號(hào)
? 恢復(fù)原信號(hào)的操作
trap -p
? 列出自定義信號(hào)操作
trap finish EXIT
? 當(dāng)腳本退出時(shí),執(zhí)行finish函數(shù)
- 變量:存儲(chǔ)單個(gè)元素的內(nèi)存空間
- 數(shù)組:存儲(chǔ)多個(gè)元素的連續(xù)的內(nèi)存空間,相當(dāng)于多個(gè)變量的集合
- 數(shù)組名和索引
?索引:編號(hào)從0開始,屬于數(shù)值索引
?注意:索引可支持使用自定義的格式,而不僅是數(shù)值格式,即為關(guān)聯(lián)索引,bash5.0版本之后開始支持
?bash的數(shù)組支持稀疏格式(索引不連續(xù))- 聲明數(shù)組:
?declare -a ARRAY_NAME
?declare -A ARRAY_NAME 關(guān)聯(lián)數(shù)組
?注意:兩者不可相互轉(zhuǎn)換
- 數(shù)組元素的賦值
?(1) 一次只賦值一個(gè)元素
???ARRAY_NAME[INDEX]=VALUE
???weekdays[0]="Sunday"
???weekdays[4]="Thursday"
?(2) 一次賦值全部元素
???ARRAY_NAME=("VAL1" "VAL2" "VAL3" ...)
?(3) 只賦值特定元素
???ARRAY_NAME=([0]="VAL1" [3]="VAL2" ...)
?(4) 交互式數(shù)組值對(duì)賦值
???read -a ARRAY
declare -a
- 引用數(shù)組元素
?${ARRAY_NAME[INDEX]}
?注意:省略[INDEX]表示引用下標(biāo)為0的元素- 引用數(shù)組所有元素
?${ARRAY_NAME[]}
?${ARRAY_NAME[@]}- 數(shù)組的長(zhǎng)度(數(shù)組中元素的個(gè)數(shù))
?${#ARRAY_NAME[]}
?${#ARRAY_NAME[@]}- 刪除數(shù)組中的某元素:導(dǎo)致稀疏格式
?unset ARRAY[INDEX]- 刪除整個(gè)數(shù)組
?unset ARRAY
- 引用數(shù)組中的元素:??數(shù)組切片:
???${ARRAY[@]:offset:number}
br/>??數(shù)組切片:
???${ARRAY[@]:offset:number}
?????number 要取出的元素個(gè)數(shù)
???取偏移量之后的所有元素
???${ARRAY[@]:offset}- 向數(shù)組中追加元素:
?ARRAY[${#ARRAY[*]}]=value- 關(guān)聯(lián)數(shù)組:
?declare -A ARRAY_NAME
?ARRAY_NAME=([idx_name1]='val1' [idx_name2]='val2‘...)
?注意:關(guān)聯(lián)數(shù)組必須先聲明再調(diào)用
- ${#var}:返回字符串變量var的長(zhǎng)度
- ${var:offset}:返回字符串變量var中從第offset個(gè)字符后(不包括第offset個(gè)字符)的字符開始,到最后的部分,offset的取值在0 到 ${#var}-1 之間(bash5.2后,允許為負(fù)值)
- ${var:offset:number}:返回字符串變量var中從第offset個(gè)字符后(不包括第offset個(gè)字符)的字符開始,長(zhǎng)度為number的部分
- ${var: -length}:取字符串的最右側(cè)幾個(gè)字符
?注意:冒號(hào)后必須有一空白字符- ${var:offset:-length}:從最左側(cè)跳過offset字符,一直向右取到距離最右側(cè)lengh個(gè)字符之前的內(nèi)容
- ${var: -length:-offset}:先從最右側(cè)向左取到length個(gè)字符開始,再向右取到距離最右側(cè)offset個(gè)字符之間的內(nèi)容
?注意:-length前空格
- 基于模式取子串
?${var#word}:其中word可以是指定的任意字符
?功能:自左而右,查找var變量所存儲(chǔ)的字符串中,第一次出現(xiàn)的word, 刪除字符串開頭至第一次出現(xiàn)word字符串(含)之間的所有字符
?${var##word}:同上,貪婪模式,不同的是,刪除的是字符串開頭至最后一次由word指定的字符之間的所有內(nèi)容- 示例:
?file=“var/log/messages”
?${file#/}: log/messages
?${file##/}: messages- ${var%word}:其中word可以是指定的任意字符
?功能:自右而左,查找var變量所存儲(chǔ)的字符串中,第一次出現(xiàn)的word, 刪除字符串最后一個(gè)字符向左至第一次出現(xiàn)word字符串(含)之間的所有字符
?file="/var/log/messages"
?${file%/}: /var/log- ${var%%word*}:同上,只不過刪除字符串最右側(cè)的字符向左至最后一次出現(xiàn)word字符之間的所有字符
- 示例:
?url=http://www.magedu.com:80
?${url##:} 80
?${url%%:} http- 查找替換
?${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為行尾所匹配到的字符串- 字符大小寫轉(zhuǎn)換
?${var^^}:把var中的所有小寫字母轉(zhuǎn)換為大寫
?${var,,}:把var中的所有大寫字母轉(zhuǎn)換為小寫
- Shell變量一般是無類型的,但是bash Shell提供了declare和typeset兩個(gè)命令用于指定變量的類型,兩個(gè)命令是等價(jià)的
declare [選項(xiàng)] 變量名
??-r 聲明或顯示只讀變量
?-i 將變量定義為整型數(shù)
?-a 將變量定義為數(shù)組
?-A 將變量定義為關(guān)聯(lián)數(shù)組
?-f 顯示已定義的所有函數(shù)名及其內(nèi)容
?-F 僅顯示已定義的所有函數(shù)名
?-x 聲明或顯示環(huán)境變量和函數(shù)
?-l 聲明變量為小寫字母 declare –l var=UPPER
?-u 聲明變量為大寫字母 declare –u var=lowereval命令- eval命令將會(huì)首先掃描命令行進(jìn)行所有的置換,然后再執(zhí)行該命令。該命令適用于那些一次掃描無法實(shí)現(xiàn)其功能的變量.該命令對(duì)變量進(jìn)行兩次掃描
示例:
?[root@server ~]# CMD=whoami
?[root@server ~]# echo $CMD
?whoami
?[root@server ~]# eval $CMD
?root
?[root@server ~]# n=10
?[root@server ~]# echo {0..$n}
?{0..10}
?[root@server ~]# eval echo {0..$n}
?0 1 2 3 4 5 6 7 8 9 10
- 如果第一個(gè)變量的值是第二個(gè)變量的名字,從第一個(gè)變量引用第二個(gè)變量的值就稱為間接變量引用
- variable1的值是variable2,而variable2又是變量名,variable2的值為value,間接變量引用是指通過variable1獲得變量值value的行為
?variable1=variable2
?variable2=value- bash Shell提供了兩種格式實(shí)現(xiàn)間接變量引用
?eval tempvar=\$$variable1
?tempvar=${!variable1}
- mktemp命令:創(chuàng)建并顯示臨時(shí)文件,可避免沖突
- mktemp [OPTION]... [TEMPLATE]
?TEMPLATE: filenameXXX
???X至少要出現(xiàn)三個(gè)- OPTION:
?-d: 創(chuàng)建臨時(shí)目錄
?-p DIR或--tmpdir=DIR:指明臨時(shí)文件所存放目錄位置- 示例:
?mktemp /tmp/testXXX
?tmpdir=mktemp –d /tmp/testdirXXX
?mktemp --tmpdir=/testdir testXXXXXX
??expect [選項(xiàng)] [ -c cmds ] [ [ -[f|b] ] cmdfile ] [ args ]
-c:從命令行執(zhí)行expect腳本,默認(rèn)expect是交互地執(zhí)行的
???示例:expect -c 'expect "\n" {send "pressed enter\n"}
-d:可以輸出輸出調(diào)試信息
???示例:expect -d ssh.exp
??spawn 啟動(dòng)新的進(jìn)程
?send 用于向進(jìn)程發(fā)送字符串
?expect 從進(jìn)程接收字符串
?interact 允許用戶交互
?exp_continue 匹配多個(gè)字符串在執(zhí)行動(dòng)作后加此命令
?set timeout 等待命令輸出時(shí)間,默認(rèn)是10秒。一旦到了這個(gè)timeout,還是沒有屏幕輸出的話,expect腳本中下面的代碼就會(huì)執(zhí)行
單一分支模式語法:
??expect “hi” {send “You said hi\n"}
?匹配到hi后,會(huì)輸出“you said hi” ,并換行多分支模式語法:
??expect "hi" { send "You said hi\n" } \
???"hehe" { send "Hehe yourself\n" } \
???"bye" { send "Good bye\n" }
?匹配hi,hello,bye任意字符串時(shí),執(zhí)行相應(yīng)輸出。等同如下:
?expect {
???"hi" { send "You said hi\n"}
???"hehe" { send "Hehe yourself\n"}
???"bye" { send " Good bye\n"}
?}
.
在識(shí)別字符串時(shí) " 與 { 之間一定要有空格,否則不會(huì)識(shí)別成功,切記?。?!
??#!/usr/bin/expect
?spawn scp /etc/fstab 192.168.8.100:/app
?expect {
???"yes/no" { send "yes\n";exp_continue }
???"password" { send "magedu\n" }
?}
?expect eof