16.循環執行
將代碼段重復運行多次
重復運行多少次;
循環次數事先已知
循環次數事先未知
有進入條件和退出條件
for,while,until
for循環
for 變量名 in 列表;do
循環體
done
執行機制:
依次將列表中的元素賦值給“變量名”;每次賦值后即執行一次循環體;直到列表中的元素耗盡,循環結束
列表生成方式:
1.直接給出列表
2.整數列表:
(a){1..10}
(b) $(seq 1 10)
3.返回列表的命令
$(command)
4.使用glob,如*.sh
5.變量引用:$@,$*
案例練習:
1、判斷/var/目錄下所有文件的類型
#!/bin/bash # a=`echo /var/ | sed -r 's@(.*)/@\1@'` for b in `ls -A /var/` do c=`ls -dl $a/$b | cut -c1` case $c in f) echo "$b iWJ" ;; d) echo "$b ML" ;; *) echo "$b NO ZD" esac done
2、添加10個用戶user1-user10,密碼同用戶名
#!/bin/bash # u=user1 for i in {1..10};do if id -u $u$i &>/dev/null;then echo "$u$i CZ" else useradd $u$i &> /dev/null echo "$u$i useradd success!!" echo $u$i |passwd --stdin $u$i &> /dev/null echo "$u$i passwd success!!" fi done
3、/etc/rc.d/rc3.d目錄下分別有多個以K開頭和以S開頭的 文件;分別讀取每個文件,以K開頭的文件輸出為文件加stop ,以S開頭的文件輸出為文件名加start; “K34filename stop” “S66filename start”
#!/bin/bash # file=`ls -A /etc/rc.d/rc3.d/` for i in $file ;do s=`echo $i | cut -c1` case $s in K) echo "$i stop" ;; S) echo "$i start" ;; esac done
4、寫一個腳本,提示輸入正整數n的值,計算1+2+3+…n的 總和
#!/bin/bash # read -p "SRSZ:" n #m=`echo "$n" | grep -E "^[[:digit:]]+"` if [ $n -eq 0 ];then echo "zheshi 0" fi for i in `seq $n` do sum=$(($sum+$i)) done echo `seq -s + 1 $n`=$sum
5、寫一個腳本,提示請輸入網絡地址,如192.168.0.0,判 斷輸入的網段中主機在線狀態
#!/bin/bash # read -p "shuru IP:" ip num=`echo $ip |sed -r 's@(.*).$@\1@'` if `echo $ip |egrep '\<([1-9]|[1-9][0-9]|1[0-9][0-9]|2[0-4][0-9]|25[0-5])\.(([0-9]|[1-9][0-9]|1[0-9][0-9]|2[0-4][0-9]|25[0-5])\.){2}([0-9]|[1-9][0-9]|1[0-9][0-9]|2[0-4][0-9]|25[0-5])\>' &>/dev/null` then for i in `seq 113 116` do ping -c1 -W2 $num$i &>/dev/null && echo "$num$i success!" || echo "$num$i error" done else echo "$ip bibi" fi
6、打印九九乘法表
#!/bin/bash # for i in `seq 9` do for b in `seq 1 $i` do echo -ne "$b*$i=$(($b*$i))\t" done echo done echo = = = = = = = = = = = = = = 2 = = = = = = = = = = = = = for ((i=1; i<=9; i++)) do for ((j=1; j<=i; j++)) do echo -ne "$j*$i=$(($j*$i))\t" done echo done
while 循環
while CONDITION;do
循環體
done
CONDITION:循環控制條件;進入循環之前,先做一次判斷,每一次循環之后會在再次做判斷,條件為“true”則執行一次循環,直到條件測試狀態為false終止循環
CONDITION:一般應該有循環控制變量;而此變量的值會在循環體不斷地被修正
進入條件:CONDITION為true;
退出條件:CONDITION為false;
案例練習:
1、求100以內所有正整數之和
#!/bin/bash # declare -i SUM=0 declare -i I=0 while [ $I -le 100 ] do SUM+=$I let I++ done echo "$SUM"
2、通過ping命令探測172.16.250.1-254范圍內的所有主機 的在線狀態,統計在線主機和離線主機各多少。
#!/bin/bash # ip=172.16.250. i=1 c=0 s=0 while [ $i -lt 255 ];do if ping -c1 -W2 $ip$i &>/dev/null;then echo $ip$i sccuess let c++ else echo $ip$i error let s++ fi let i++ done echo sccuess:$c echo error:$s
3、打印九九乘法表
#!/bin/bash # declare -i i=1 while [ $i -le 9 ];do declare -i j=1 while [ $j -le $i ];do echo -ne "$j*$i=$(($j*$i))\t" let j++ done echo let i++ done
4、利用變量RANDOM生成10個隨機數字,輸出這個10數字 ,并顯示其中的最大者和最小者
#!/bin/bash # declare -i i=0 q=`while [ $i -lt 10 ];do s=$[$RANDOM] echo $s let i++ done` b=`echo $q |tr " " "\n"|sort -n |head -1` v=`echo $q |tr " " "\n"|sort -n |tail -1` echo -e "D:$b\nX:$v"
5、打印國際象棋棋盤
#!/bin/bash # declare -i i=1 while [ $i -le 8 ];do declare -i j=1 while [ $j -le 8 ];do [ $[$i%2-$j%2] -eq 0 ] && echo -ne "\033[41m \033[0m" ||echo -ne "\033[47m \033[0m" let j++ done echo let i++ done
until循環
until循環就是和while相反的,只需要把while的思路反過來即可
until CONDITION;do
循環體
done
進入條件:CONDITION為false
退出條件:CONDITION為true
循環控制語句continue
用于循環體重
continue[N]:提前結束第N層的本輪循環,而直接進入下一輪判斷;最內層為第一層
while CONDITION1;do
CMD1
…
if CONDITION2;then
CONDITION
fi
CMDN
….
done
循環控制語句break
用于循環體中
break[N]:提前結束第N層循環,最內層為第1層
while CONDITION1;do
CMD1
…
if CONDITION2;then
break
fi
CMDn
…
done
創建無線循環
while true;do
循環體
done
until false;do
循環體
done
案例練習:
1、每隔3秒鐘到系統上獲取已經登錄的用戶的信息;如果發 現用戶hacker登錄,則將登錄時間和主機記錄于日志 /var/log/login.log中,并提示該用戶退出系統。
#!/bin/bash # while true;do who | grep hacker &> /dev/null if [ $? -eq 0 ];then who >> /var/log/login.log echo "hacker login exit!!!!" exit sleep 3 fi done
2、隨機生成10以內的數字,實現猜字游戲,提示比較大或小 ,相等則退出。
#!/bin/bash n=$[$RANDOM%10+1] i=0 echo "猜數字游戲開始啦!" while true do read -p "shuru:" in let i++ if [ $in -eq $n ];then echo "猜對嘍! $n" echo "一共猜了$i次" exit 0 elif [ $in -gt $n ];then echo "有點高了,數字小一點試試" else echo "有點小了,數字大一點試試" fi done
特殊用法
while循環的特殊用法(遍歷文件的每一行)
while read line;do
循環體
done < /PATH/FROM/SOMEFILE
依次讀取/PATH/FROM/SOMEFILE文件中的每一行,且將每行賦值給變量line
案例練習:
1.掃描/etc/passwd文件每一行,如發現GECOS字段為空,則填 充用戶名和單位電話為62985600,并提示該用戶的GECOS信 息修改成功
#!/bin/bash # file="/etc/passwd" while read line do u=$(echo $line |cut -d: -f1) g=$(echo $line |cut -d: -f5) chfn -f $u -p 62985600 $u &> /dev/null if [ -z $g ];then echo "$u GECOS sccuess" else echo "$u NOweikong" #usermod -c "" $u fi done < $file
特殊用法:
雙小括號方法:即((…))格式,也可以用于算術運算
雙小括號方法也可以使bash shell 實現c語言風格的變量操作
#I=10
#((I++))
for循環的特殊格式:
for((控制變量初始化;條件判斷表達式;控制變量的修正表達式))
do
循環體
done
控制變量初始化:僅在運行到循環代碼段時執行一次
控制變量的修正表達式:每輪循環結束會先進行控制變量修正運算,而后在做條件判斷
這種用法就像我們上面寫的99乘法表一樣,使用的是c語言風格的
#!/bin/bash # for ((i=1; i<=9; i++)) do for ((j=1; j<=i; j++)) do echo -ne "$j*$i=$(($j*$i))\t" done echo done
select循環與菜單
select variable in list
do
循環體命令
done
select循環主要用戶創建菜單,按數字順序排列的菜單項將顯示在標準錯誤上,并顯示ps3提示符,等待用戶輸入。
用戶輸入菜單列表中的某個數字,執行相應的命令。
用戶輸入被保存在內置變量REPLY中
select 與 case
select是個無限循環,因此要記住用break命令退出循環,或用exit命令終止命令,也可以按ctrl+c來退出循環。
select經常和case聯合使用
與for循環類似,可以省略 in list,此時使用位置參量。
函數介紹
函數function是由若干條shell命令組的語句塊,實現代碼重用和模塊化編程。
它與shell程序形式上是相似的,不同的是它不是一個單獨的 進程,不能獨立運行,而是shell程序的一部分。
函數和shell程序比較相似,區別在于:
Shell程序在子Shell中運行
而Shell函數在當前Shell中運行。因此在當前Shell中,函數可以 對shell中變量進行修改
定義函數
函數由兩部分組成:函數名和函數體。
語法一:
function f_name {
…函數體…
}
語法二:
f_name() {
…函數體…
}
函數的使用
函數的定義和使用:
可在交互式環境下定義函數
可將函數放在腳本文件中作為它的一部分
可放在只包含函數的單獨文件中
調用:函數只有被調用才會執行;
調用:給定函數名
函數名出現的地方,會被自動替換為函數代碼
函數的生命周期:被調用時創建,返回時終止
函數返回值
函數一共有兩種返回值:
函數的執行結果返回值:
(1) 使用echo或printf命令進行輸出
(2) 函數體中調用命令的輸出結果
函數的退出狀態碼:
(1) 默認取決于函數中執行的最后一條命令的退出狀態碼
(2) 自定義退出狀態碼,其格式為:
return 從函數中返回,用最后狀態命令決定返回值
return 0 無錯誤返回。
return 1-255 有錯誤返回
交互式環境下定義和使用函數
示例:
$dir() {
> ls -l
> }
定義該函數后,若在$后面鍵入dir,其顯示結果同ls -l的 作用相同。
$dir
該dir函數將一直保留到用戶從系統退出,或執行了如下 所示的unset命令: $ unset dir
在腳本中定義及使用函數
函數在使用前必須定義,因此應將函數定義放在腳本開始部分,直至 shell首次發現它后才能使用
調用函數僅使用其函數名即可。
示例: $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"
使用函數文件
可以將經常使用的函數存入函數文件,然后將函 數文件載入shell。
文件名可任意選取,但最好與相關任務有某種聯 系。例如:functions.main
一旦函數文件載入shell,就可以在命令行或腳本 中調用函數??梢允褂胹et命令查看所有定義的函 數,其輸出列表包括已經載入shell的所有函數。
若要改動函數,首先用unset命令從shell中刪除 函數。改動完畢后,再重新載入此文件。
函數文件示例:
#!/bin/bash #functions.main findit() { if [ $# -lt 1 ] ; then echo "Usage:findit file" return 1 fi find / -name $1 –print }
載入函數:
函數文件已創建好后,要將它載入shell
定位函數文件并載入shell的格式:. filename 或 source filename
注意:此即<點> <空格> <文件名> 這里的文件名要帶正確路徑
示例:上例中的函數,可使用如下命令: $ . functions.main
檢查載入函數:
使用set命令檢查函數是否已載入。set命令將在shell中顯示 所有的載入函數。
示例:
$set findit=( ) { if [ $# -lt 1 ]; then echo "usage :findit file"; return 1 fi find / -name $1 -print }
執行shell函數
要執行函數,簡單地鍵入函數名即可:
示例:
$findit groups
/usr/bin/groups
/usr/local/backups/groups.bak
刪除shell函數
現在對函數做一些改動。首先刪除函數,使其對shell不可用 。使用unset命令完成此功能.
命令格式為: unset function_name
示例:$unset findit 再鍵入set命令,函數將不再顯示
函數參數:
函數可以接受參數:
傳遞參數給函數:調用函數時,在函數名后面以空白分隔 給定參數列表即可;例如“testfunc arg1 arg2 …”
在函數體中當中,可使用$1, $2, …調用這些參數;還 可以使用$@, $*, $#等特殊變量 函數變量
變量作用域:
環境變量:當前shell和子shell有效
本地變量:只在當前shell進程有效,為執行腳本會啟動
專用子shell進程;因此,本地變量的作用范圍是當前shell腳本 程序文件,包括腳本中的函數。
局部變量:函數的生命周期;函數結束時變量被自動銷毀
注意:如果函數中有局部變量,如果其名稱同本地變量,使 用局部變量。
在函數中定義局部變量的方法 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)!
函數示例:
#!/bin/bash # fact() { if [ $1 -eq 0 -o $1 -eq 1 ]; then echo 1 else echo $[$1*$(fact $[$1-1])] fi } fact $1
案例練習:
1、編寫服務腳本/root/bin/testsrv.sh,完成如下要求
(1) 腳本可接受參數:start, stop, restart, status
(2) 如果參數非此四者之一,提示使用格式后報錯退出
(3) 如是start:則創建/var/lock/subsys/SCRIPT_NAME, 并顯示“啟動成功” 考慮:如果事先已經啟動過一次,該如何處理?
(4) 如是stop:則刪除/var/lock/subsys/SCRIPT_NAME, 并顯示“停止完成” 考慮:如果事先已然停止過了,該如何處理?
(5) 如是restart,則先stop, 再start 考慮:如果本來沒有start,如何處理?
(6) 如是status, 則如果/var/lock/subsys/SCRIPT_NAME文件存在,則顯示 “SCRIPT_NAME is running…” 如果/var/lock/subsys/SCRIPT_NAME文件不存在,則顯示“SCRIPT_NAME is stopped…” 其中:SCRIPT_NAME為當前腳本名
(7)在所有模式下禁止啟動該服務,可用chkconfig 和 service命令管理
#!/bin/bash # file=$(basename $0) SCRIPTS="/var/lock/subsys/$file" function start () { if [ ! -e $SCRIPTS ];then touch $SCRIPTS sleep 1 echo "service $file start sccuess!" else echo "service $file running....." fi } function stop () { if [ -e $SCRIPTS ];then rm -rf $SCRIPTS sleep 1 echo "service $file stop sccuess!" else echo "service $file status stop!" fi } function status () { if [ -e $SCRIPTS ];then echo "$file is running..." else echo "$file is stopped..." fi } case $1 in start) start ;; stop) stop ;; restart) stop start ;; status) status ;; *) echo "$file usage {start|stop|status|restart}" exit 1 ;; esac
2、編寫腳本/root/bin/copycmd.sh
(1) 提示用戶輸入一個可執行命令名稱
(2) 獲取此命令所依賴到的所有庫文件列表
(3) 復制命令至某目標目錄(例如/mnt/sysroot)下的對應路徑下; 如:/bin/bash ==> /mnt/sysroot/bin/bash /usr/bin/passwd ==> /mnt/sysroot/usr/bin/passwd
(4) 復制此命令依賴到的所有庫文件至目標目錄下的對應路徑下: 如:/lib64/ld-linux-x86-64.so.2 ==> /mnt/sysroot/lib64/ldlinux-x86-64.so.2
(5)每次復制完成一個命令后,不要退出,而是提示用戶鍵入新的要復制的命 令,并重復完成上述功能;直到用戶輸入quit退出
#!/bin/bash # file=/mnt/sysroot/ libfile=/mnt/sysroot/lib64/ copycmd(){ [ -e /mnt/sysroot ] || mkdir /mnt/sysroot cmd=`which --skip-alias $p` cp $cmd $file && echo "Copy command success." } copylib(){ lib=$(ldd `which --skip-alias $p` |grep -E -o "/.*" | cut -d" " -f1) cp $lib $libfile [ $? -eq 0 ] && echo "Copy lib success." } while true do read -p " 請輸入命令:" p if [[ $p == "quit" ]];then exit fi if which --skip-alias $p &> /dev/null ;then copycmd copylib else read -p " 不是命令, 請輸入命令:" p fi done
原創文章,作者:鬧鐘哥,如若轉載,請注明出處:http://www.www58058.com/39824