Shell 腳本進(jìn)階、經(jīng)典用法及其案例
一、條件選擇、判斷
1、條件選擇if
(1)用法格式
- if 判斷條件 1 ; then
 - 條件為真的分支代碼
 - elif 判斷條件 2 ; then
 - 條件為真的分支代碼
 - elif 判斷條件 3 ; then
 - 條件為真的分支代碼
 - else
 - 以上條件都為假的分支代碼
 - fi
 
逐條件進(jìn)行判斷,第一次遇為“真”條件時(shí),執(zhí)行其分支,而后結(jié)束整個(gè)if。
(2)經(jīng)典案例:
- #判斷年紀(jì)
 - #!/bin/bash
 - read -p "Please input your age: " age
 - if [[ $age =~ [^0-9] ]] ;then
 - echo "please input a int"
 - exit 10
 - elif [ $age -ge 150 ];then
 - echo "your age is wrong"
 - exit 20
 - elif [ $age -gt 18 ];then
 - echo "good good work,day day up"
 - else
 - echo "good good study,day day up"
 - fi
 
分析:請(qǐng)輸入年紀(jì),先判斷輸入的是否含有除數(shù)字以外的字符,有,就報(bào)錯(cuò);沒有,繼續(xù)判斷是否小于150,是否大于18。
- #判斷分?jǐn)?shù)
 - #!/bin/bash
 - read -p "Please input your score: " score
 - if [[ $score =~ [^0-9] ]] ;then
 - echo "please input a int"
 - exit 10
 - elif [ $score -gt 100 ];then
 - echo "Your score is wrong"
 - exit 20
 - elif [ $score -ge 85 ];then
 - echo "Your score is very good"
 - elif [ $score -ge 60 ];then
 - echo "Your score is soso"
 - else
 - echo "You are loser"
 - fi
 
分析:請(qǐng)輸入成績(jī),先判斷輸入的是否含有除數(shù)字以外的字符,有,就報(bào)錯(cuò);沒有,繼續(xù)判斷是否大于100,是否大于85,是否大于60。
2、條件判斷 case
(1)用法格式
- case $name in;
 - PART1)
 - cmd
 - ;;
 - PART2)
 - cmd
 - ;;
 - *)
 - cmd
 - ;;
 - esac
 
- 注意:case 支持glob 風(fēng)格的通配符:
 - *: 任意長(zhǎng)度任意字符
 - ?: 任意單個(gè)字符
 - [] :指定范圍內(nèi)的任意單個(gè)字符
 - a|b: a 或b
 
(2)案例:
- #判斷yes or no
 - #!/bin/bash
 - read -p "Please input yes or no: " anw
 - case $anw in
 - [yY][eE][sS]|[yY])
 - echo yes
 - ;;
 - [nN][oO]|[nN])
 - echo no
 - ;;
 - *)
 - echo false
 - ;;
 - esac
 
分析:請(qǐng)輸入yes or no,回答Y/y、yes各種大小寫組合為yes;回答N/n、No各種大小寫組合為no。
二、四個(gè)循環(huán)
1、for
(1)用法格式
- ① for name in 列表 ;do
 - 循環(huán)體
 - done
 - ② for (( exp1; exp2; exp3 )) ;do
 - cmd
 - done
 
exp1只執(zhí)行一次,相當(dāng)于在for里嵌了while
③ 執(zhí)行機(jī)制:
- 依次將列表中的元素賦值給“變量名”; 每次賦值后即執(zhí)行一次循環(huán)體; 直到列表中的元素耗盡,循環(huán)結(jié)束
 - 列表的表示方法,可以glob 通配符,如{1..10} 、*.sh ;也可以變量引用,如:seq 1 $name
 
(2)案例
- #求出(1+2+...+n)的總和
 - sum=0
 - read -p "Please input a positive integer: " num
 - if [[ $num =~ [^0-9] ]] ;then
 - echo "input error"
 - elif [[ $num -eq 0 ]] ;then
 - echo "input error"
 - else
 - for i in `seq 1 $num` ;do
 - sum=$[$sum+$i]
 - done
 - echo $sum
 - fi
 - unset zhi
 
分析:sum初始值為0,請(qǐng)輸入一個(gè)數(shù),先判斷輸入的是否含有除數(shù)字以外的字符,有,就報(bào)錯(cuò);沒有判斷是否為0,不為0進(jìn)入for循環(huán),i的范圍為1~輸入的數(shù),每次的循環(huán)為sum=sum+i,循環(huán)結(jié)束,最后輸出sum的值。
- #求出(1+2+...+100)的總和
 - for (( i=1,num=0;i<=100;i++ ));do
 - [ $[i%2] -eq 1 ] && let sum+=i
 - done
 - echo sum=$sum
 - 分析:i=1,num=0;當(dāng)i<=100,進(jìn)入循環(huán),若i÷2取余=1,則sumsum=sum+i,ii=i+1。
 
2、while
(1)用法格式
- while 循環(huán)控制條件 ;do
 - 循環(huán)
 - done
 
循環(huán)控制條件;進(jìn)入循環(huán)之前,先做一次判斷;每一次循環(huán)之后會(huì)再次做判斷;條件為“true” ,則執(zhí)行一次循環(huán);直到條件測(cè)試狀態(tài)為“false” 終止循環(huán)
(2)特殊用法(遍歷文件的每一行):
- while read line; do控制變量初始化
 - 循環(huán)體
 - done < /PATH/FROM/SOMEFILE
 - 或cat /PATH/FROM/SOMEFILE | while read line; do
 - 循環(huán)體
 - done
 
依次讀取/PATH/FROM/SOMEFILE文件中的每一行,且將行賦值給變量line
(3)案例:
- #100以內(nèi)所有正奇數(shù)之和
 - sum=0
 - i=1
 - while [ $i -le 100 ] ;do
 - if [ $[$i%2] -ne 0 ];then
 - let sum+=i
 - let i++
 - else
 - let i++
 - fi
 - done
 - echo "sum is $sum"
 
分析:sum初始值為0,i的初始值為1;請(qǐng)輸入一個(gè)數(shù),先判斷輸入的是否含有除數(shù)字以外的字符,有,就報(bào)錯(cuò);沒有當(dāng)i<100時(shí),進(jìn)入循環(huán),判斷 i÷2取余 是否不為0,不為0時(shí)為奇數(shù),sum=sum+i,i+1,為0,i+1;循環(huán)結(jié)束,最后輸出sum的值。
3、until 循環(huán)
(1)用法
- unitl 循環(huán)條件 ;do
 - 循環(huán)
 - done
 
進(jìn)入條件:循環(huán)條件為true ;退出條件:循環(huán)條件為false;剛好和while相反,所以不常用,用while就行。
(2)案例
- #監(jiān)控xiaoming用戶,登錄就殺死
 - until pgrep -u xiaoming &> /dev/null ;do
 - sleep 0.5
 - done
 - pkill -9 -u xiaoming
 
分析:每隔0.5秒掃描,直到發(fā)現(xiàn)xiaoming用戶登錄,殺死這個(gè)進(jìn)程,退出腳本,用于監(jiān)控用戶登錄。
4、select 循環(huán)與菜單
(1)用法
- 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í)使用位置參量
 
(2)案例
- #生成菜單,并顯示選中的價(jià)錢
 - PS3="Please choose the menu: "
 - select menu in mifan huimian jiaozi babaozhou quit
 - do
 - case $REPLY in
 - 1|4)
 - echo "the price is 15"
 - ;;
 - 2|3)
 - echo "the price is 20"
 - ;;
 - 5)
 - break
 - ;;
 - *)
 - echo "no the option"
 - esac
 - done
 
分析:PS3是select的提示符,自動(dòng)生成菜單,選擇5break退出循環(huán)。
三、循環(huán)里的一些用法
1、循環(huán)控制語句
(1)語法
continue [N]:提前結(jié)束第N層的本輪循環(huán),而直接進(jìn)入下一輪判斷;最內(nèi)層為第1層 break [N]:提前結(jié)束第N層循環(huán),最內(nèi)側(cè)為第1層
- 例:while CONDTITON1; do
 - CMD1
 - if CONDITION2; then
 - continue / break
 - fi
 - CMD2
 - done
 
(2)案例:
- #①求(1+3+...+49+53+...+100)的和
 - #!/bin/bash
 - sum=0
 - for i in {1..100} ;do
 - [ $i -eq 51 ] && continue
 - [ $[$i%2] -eq 1 ] && { let sum+=i;let i++; }
 - done
 - echo sum=$sum
 
分析:做1+2+...+100的循環(huán),當(dāng)i=51時(shí),跳過這次循環(huán),但是繼續(xù)整個(gè)循環(huán),結(jié)果為:sum=2449
- #②求(1+3+...+49)的和
 - #!/bin/bash
 - sum=0
 - for i in {1..100} ;do
 - [ $i -eq 51 ] && break
 - [ $[$i%2] -eq 1 ] && { let sum+=i;let i++; }
 - done
 - echo sum=$sum
 
分析:做1+2+...+100的循環(huán),當(dāng)i=51時(shí),跳出整個(gè)循環(huán),結(jié)果為:sum=625
2、循環(huán)控制shift命令
(1)作用
用于將參數(shù)列表list左移指定次數(shù),最左端的那個(gè)參數(shù)就從列表中刪除,其后邊的參數(shù)繼續(xù)進(jìn)入循環(huán)
(2)案例:
- #①創(chuàng)建指定的多個(gè)用戶
 - #!/binbash
 - if [ $# -eq 0 ] ;then
 - echo "Please input a arg(eg:`basename $0` arg1)"
 - exit 1
 - else
 - while [ -n "$1" ];do
 - useradd $1 &> /dev/null
 - shift
 - done
 - fi
 
分析:如果沒有輸入?yún)?shù)(參數(shù)的總數(shù)為0),提示錯(cuò)誤并退出;反之,進(jìn)入循環(huán);若第一個(gè)參數(shù)不為空字符,則創(chuàng)建以第一個(gè)參數(shù)為名的用戶,并移除第一個(gè)參數(shù),將緊跟的參數(shù)左移作為第一個(gè)參數(shù),直到?jīng)]有第一個(gè)參數(shù),退出。
- #②打印直角三角形的字符
 - #!/binbash
 - while (( $# > 0 ))
 - do
 - echo "$*"
 - shift
 - done
 
3、返回值結(jié)果
- true 永遠(yuǎn)返回成功結(jié)果
 - : null command ,什么也不干,返回成功結(jié)果
 - false 永遠(yuǎn)返回錯(cuò)誤結(jié)果
 
創(chuàng)建無限循環(huán)
- while true ;do
 - 循環(huán)體
 - done
 
4、循環(huán)中可并行執(zhí)行,使腳本運(yùn)行更快
(1)用法
- for name in 列表 ;do
 - {
 - 循環(huán)體
 - }&
 - done
 - wait
 
(2)實(shí)例:
- #搜尋自己指定ip(子網(wǎng)掩碼為24的)的網(wǎng)段中,UP的ip地址
 - read -p "Please input network (eg:192.168.0.0): " net
 - echo $net |egrep -o "\<(([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\.){3}([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\>"
 - [ $? -eq 0 ] || ( echo "input error";exit 10 )
 - IP=`echo $net |egrep -o "^([0-9]{1,3}\.){3}"`
 - for i in {1..254};do
 - {
 - ping -c 1 -w 1 $IP$i &> /dev/null && \
 - echo "$IP$i is up"
 - }&
 - done
 - wait 分析:請(qǐng)輸入一個(gè)IP地址例192.168.37.234,如果格式不是0.0.0.0 則報(bào)錯(cuò)退出;正確則進(jìn)入循環(huán),IP變量的值為192.168.37. i的范圍為1-254,并行ping 192.168.37.1-154,ping通就輸出此IP為UP。直到循環(huán)結(jié)束。
 
四、信號(hào)捕獲trap
1、用法格式
- trap ' 觸發(fā)指令' 信號(hào),自定義進(jìn)程收到系統(tǒng)發(fā)出的指定信號(hào)后,將執(zhí)行觸發(fā)指令,而不會(huì)執(zhí)行原操作
 - trap '' 信號(hào),忽略信號(hào)的操作
 - trap '-' 信號(hào),恢復(fù)原信號(hào)的操作
 - trap -p,列出自定義信號(hào)操作
 
信號(hào)可以3種表達(dá)方法:信號(hào)的數(shù)字2、全名SIGINT、縮寫INT
2、常用信號(hào)
- 1) SIGHUP: 無須關(guān)閉進(jìn)程而讓其重讀配置文件
 - 2) SIGINT: 中止正在運(yùn)行的進(jìn)程;相當(dāng)于Ctrl+c
 - 3) SIGQUIT: 相當(dāng)于ctrl+\
 - 9) SIGKILL: 強(qiáng)制殺死正在運(yùn)行的進(jìn)程
 - 15) SIGTERM :終止正在運(yùn)行的進(jìn)程(默認(rèn)為15)
 - 18) SIGCONT :繼續(xù)運(yùn)行
 - 19) SIGSTOP :后臺(tái)休眠
 - 9 信號(hào),強(qiáng)制殺死,捕獲不住
 
3、案例
- #①打印0-9,ctrl+c不能終止
 - #!/bin/bash
 - trap 'echo press ctrl+c' 2
 - for ((i=0;i<10;i++));do
 - sleep 1
 - echo $i
 - done
 
分析:i=0,當(dāng)i<10,每休眠1秒,i+1,捕獲2信號(hào),并執(zhí)行echo press ctrl+c
- #②打印0-3,ctrl+c不能終止,3之后恢復(fù),能終止
 - #!/bin/bash
 - trap '' 2
 - trap -p
 - for ((i=0;i<3;i++));do
 - sleep 1
 - echo $i
 - done
 - trap '-' SIGINT
 - for ((i=3;i<10;i++));do
 - sleep 1
 - echo $i
 - done
 
分析:i=0,當(dāng)i<3,每休眠1秒,i+1,捕獲2信號(hào);i>3時(shí),解除捕獲2信號(hào)。
五、腳本小知識(shí)
1、生成隨機(jī)字符 cat /dev/urandom
- #生成8個(gè)隨機(jī)大小寫字母或數(shù)字
 - cat /dev/urandom |tr -dc [:alnum:] |head -c 8
 
2、生成隨機(jī)數(shù) echo $RANDOM
- 確定范圍 echo $[RANDOM%7] 隨機(jī)7個(gè)數(shù)(0-6)
 - echo $[$[RANDOM%7]+31] 隨機(jī)7個(gè)數(shù)(31-37)
 
3、echo打印顏色字
- echo -e "\033[31malong\033[0m" 顯示紅色along
 - echo -e "\033[1;31malong\033[0m" 高亮顯示紅色along
 - echo -e "\033[41malong\033[0m" 顯示背景色為紅色的along
 - echo -e "\033[31;5malong\033[0m" 顯示閃爍的紅色along
 - color=$[$[RANDOM%7]+31]
 - echo -ne "\033[1;${color};5m*\033[0m" 顯示閃爍的隨機(jī)色along
 
六、分享幾個(gè)有意思的小腳本
1、9x9乘法表
- #!/bin/bash
 - for a in {1..9};do
 - for b in `seq 1 $a`;do
 - let c=$a*$b ;echo -e "${a}x$=$c\t\c"
 - done
 - echo
 - done
 
2、彩色等腰三角形
- #!/bin/bash
 - read -p "Please input a num: " num
 - if [[ $num =~ [^0-9] ]];then
 - echo "input error"
 - else
 - for i in `seq 1 $num` ;do
 - xing=$[2*$i-1]
 - for j in `seq 1 $[$num-$i]`;do
 - echo -ne " "
 - done
 - for k in `seq 1 $xing`;do
 - color=$[$[RANDOM%7]+31]
 - echo -ne "\033[1;${color};5m*\033[0m"
 - done
 - echo
 - done
 - fi
 
3、國(guó)際象棋棋盤
- #!/bin/bash
 - red="\033[1;41m \033[0m"
 - yellow="\033[1;43m \033[0m"
 - for i in {1..8};do
 - if [ $[i%2] -eq 0 ];then
 - for i in {1..4};do
 - echo -e -n "$red$yellow";
 - done
 - echo
 - else
 - for i in {1..4};do
 - echo -e -n "$yellow$red";
 - done
 - echo
 - fi
 - done
 
























 
 
 









 
 
 
 