shell的迴圈控制語句 - continue:提前結束某次迴圈,重新開始下一次 - break:提前結束某層迴圈 範例: #求100以內的奇數和 #!/bin/bash sum=0 for i in `seq 100`;do if [ $[i%2] -ne 0 ];then continue el ...
shell的迴圈控制語句
- continue:提前結束某次迴圈,重新開始下一次
- break:提前結束某層迴圈
範例:
#求100以內的奇數和
#!/bin/bash
sum=0
for i in `seq 100`;do
if [ $[i%2] -ne 0 ];then
continue
else
let sum+=$i
fi
done
echo $sum
#實現100以內的奇數和
#!/bin/bash
sum=0
for i in `seq 100`;do #``:命令替換
if [ $[i%2] -eq 0 ];then #$[]:shell的算數語法
let sum+=$i #let工具需要藉助一個變數存儲計算後的值
fi
done
echo $sum
shell的迴圈控制 shift 命令
每執行一次shift命令,都會左移一次。
#比如有三個參數$1 $2 $3,執行一次shift命令後,$1就刪除了,$2變成$1,一次類推。
#!/bin/bash
echo $*
shift
echo $*
shift
echo $*
shift
#輸出結果:
[root@Centos8 ~]# bash shift.sh 1 2 3 4 5
1 2 3 4 5
2 3 4 5
3 4 5
#利用shift的方式批量創建賬號
#!/bin/bash
if [$# -eq 0];then #判斷位置參數是否是0
echo "請輸入需要創建的用戶"
exit
fi
while [ "$1" ] ;do #[ "$1" ]:判斷$1是否為空,只寫字元串就表示字元串不為空,為真
if id $1 &> /dev/null; then #如果id命令執行的結果為真($?=0)就表示用戶存在了
echo "$1 is exist"
else
useradd $1
echo "$1 is create"
fi
shift
done
echo "all user is create "
#使用for的形式創建賬號
if [$# -eq 0];then #判斷位置參數是否是0
echo "請輸入需要創建的用戶"
exit
fi
for user in $*;do
if id $user &> /dev/null; then #如果id命令執行的結果為真($?=0)就表示用戶存在了
echo "$user is exist"
else
useradd $user
echo "$user is create"
fi
done
echo "all user is create "
while 特殊用法 while read
文本的逐行處理。
格式
while read line; do
迴圈體
done < /PATH/FROM/SOMEFILE
while read line ;do echo $line; done #標準輸入,逐行處理
while read line ; do xxxx; done < filename #可以利用標準輸入重定向把文件傳給它
cat /etc/issue | while read line ;do echo $line; done #用管道
#使用while read判斷磁碟分區磁碟的利用率
#!/bin/bash
while true ; do #一直執行
df | sed -nr '/^\/dev\/sd/ s#([^ ]+).*([0-9]+)%.*#\1 \2#p' | while read DEV USE;do
if [ $USE -gt 80 ];then
echo "$DEV will be full, USE: $USE"
else
echo "$DEV is health, USE: $USE"
fi
done
sleep 2 #每兩秒執行一次
done
#sed -nr '/^\/dev\/sd/ s#(^[^ ]+).*([0-9]+)%.*#\1 \2#p' | while read DEV USE;do
#/^\/dev\/sd/ --- 以為/dev/sd開頭的(非空格的內容)
#[^ ]+ :取出分區的設備,非空格,一個以上的字元
#\1 \2 --- 後項替換
#查看/sbin/nologin的shell類型的用戶名和UID
#!/bin/bash
while read line ; do #通過輸入重定向的方式逐行處理內容
if [[ $line =~ /sbin/nologin$ ]];then # [[ =~ ]]:會把會把右邊的字元串當成正則
echo $line | cut -d: -f1,3 # -d:指定分隔符 -f:取那幾列
fi
done < /etc/passwd
迴圈與菜單 select
幫助我們生成菜單的作用。
格式:
#語法和for的語法很類似
select 變數 in 列表 ;do
所作的操作
done
#PS3用於輸出提示信息,效果等同於 read -p
#輸入的信息保存到置變數REPLY中
#select 是個無限迴圈,因此要用 break 命令退出迴圈
#PS1:是影響我們當前提符的
#PS2:影響多行多定向(<<)提示符的 echo $PS2--> >
#列表就是用來生成菜單的
PS3="請選擇功能(1-5):" #PS3:更改菜單的提示信息
select list in 升級軟體 備份資料庫 回滾軟體 刪庫跑路 ; do
echo $REPLY;
done
#!/bin/bash
PS3="請輸入對應選項(1-5): "
select menu in 安裝 升級 配置 卸載;do
case $REPLY in
1)
echo "install"
;;
2)
echo "update"
;;
3)
echo "config"
;;
4)
echo "remove"
;;
*)
echo "No this select"
;;
esac
done
函數:function
函數定義格式:
函數名 ()
{
函數體
}
或
function 函數
{
函數體
}
#寫了function關鍵詞,可以省略函數名後面的括弧
調用函數:直接通過函數名調用
函數名出現的地方,會被自動替換為函數代碼
[root@centos8 ~]#dir() {
> ls -l
> }
[root@centos8 ~]#dir #調用函數
total 4
函數文件:只存放函數的文件
調用函數文件
. filename 或 source filename #.和source等價
函數返回值
# 使用echo等命令進行輸出
# 函數體中調用命令的輸出結果
函數的參數
也是用$n等來實現的。
函數退出狀態碼
#1. 取決於最後一天命令的狀態碼
#2. 自定義狀態碼: retuen num #0:無錯誤返回,1-255:有錯誤返回
函數的變數
普通變數:變數名="值" #只能當前進程使用
環境變數:export 變數名="值" #當前進程和子進程都能使用
本地變數: local 變數名="值" #限定了變數在函數中有效,對外部的不影響
範例:
#判斷操作系統類型安裝軟體
#!/bin/bash
os_type
{
if cat /etc/os-release | grep -i -q "ubuntu" ; then # -q:不顯示查找的結果,不管找到與否
echo "ubuntu"
elif cat /etc/os-release | grep -E "centos" ;then
echo "centos"
else
echo "no find ubuntu or centos"
fi
}
if [ `os_type` = centos ] ;then #字元串用= != < > 等 (判斷符號兩邊需要加上空格)
yum install httpd -y
elif [ `os_type` = ubuntu ] ; then
apt install httpd -y
else
echo "nofind centos or ubuntu"
#粗略判斷ip格式是否合法
##/bin/bash
is_ip ()
{
if [[ "$1" =~ ^([0-9]{1,3}\.){3}[0-9]{1,3}$ ]] ;then
echo "$1 is valid"
else
echo "$1 is not valid"
fi
}
is_ip $1 #給函數傳遞參數
shell腳本的常用工具
shell腳本信號的捕捉:trap
trap:陷阱,用於捕獲信號的
信號的類型:
使用trap -l可以查看
CTRL+C:發送的就是2(SIGINT)信號,程式收到這個信號以後就會退出。
shell腳本捕獲信號
通過trap指令來完成:
#格式:
#通過捕獲信號來修改信號的原有功能
trap '捕獲到信號以後要做的事' 信號 #信號可以是全稱、縮寫或者前面的數字標號
例如:
trap '' 信號 #忽略信號操作
trap '-' 信號 #恢覆信號原有的功能
shell腳本創建臨時文件:mktemp
希望創建的臨時文件和現有文件不衝突
#創建臨時文件的格式
mktemp [選項] [模板] #模板就是臨時文件的文件名
#模板格式: filenameXXX,文件首碼+三個隨機字元(X(大寫)至少要出現三個)
例如;mktemp fileXXX --->file90X
#選項:
-d:創建臨時目錄文件
-p:指定創建的位置
範例
#通過創建臨時文件,實現文件垃圾箱(把不用的文件移動到一個目錄裡面)
DIR=`mktemp -d /tmp/trash-$(date +%F_%H-%M-%S)XXXXXX` #創建一個臨時目錄文件
mv $* $DIR # $*:表示所有位置的參數
echo $* is move to $DIR
安裝複製文件 install
install命令可以改許可權、拷貝、創建文件夾等
#編譯安裝make install:把make命令執行的編譯的結果複製到對應的文件中。make install 調用了install命令
#install 的用法(和cp命令很類似)
#選項
-m MODE,預設755,修改許可權
-o OWNER,指定所有者
-g GROUP,指定所有組
-d DIRNAME 目錄,創建目錄文件
##例如
install -m 770 -d /testdir/installdir #創建一個許可權為770的文件夾(install -d 相當於 mkdir -p)
install -m 700 -o tom -g root /etc/issue /etc/xx.txt #複製一個文件到指定位置,複製的時候更改了許可權
expect
把互動式的命令變成非互動式的。(espect通過對指定命令的監控,出現對應關鍵字的時候執行對應操作)
#expect中相關命令:
spawn 監控某一個命令的執行
expect 捕獲特定的字元串
send 捕獲到這個特定字元串後,發送對應信息
interact 允許用戶交互
exp_continue 匹配多個字元串在執行動作後加此命令
範例
#匹配到hi後,會輸出“you said hi”,並換行
expect "hi" {send "You said hi\n"} #捕獲屏幕上是否有hi,如果有就自動列印xx
shell腳本的數組
數組:多個變數的集合
bash的數組支持稀疏格式 #索引不連續就叫做稀疏格式
數據的申明
建議先聲明再使用。
#聲明數組:
declare -a 數據名 #申明為普通數組
declare -A 數組名 #申明為關聯數組 #管理數組必須申明後再使用,且關聯和普通不能轉換
#聲明變數
既然所有變數的預設類型是字元串型,那麼只要我們把變數聲明為整數型就可以進行運算了,
# 定義變數b並賦值為3,具有整型屬性。 declare -i x=3
數組賦值
#單個元素賦值 數組名[索引]=值
#例如:weekdays[0]="Sunday"
#一次全部賦值:數組名=(v1,v2...vn)
#範例:title=("ceo" "coo" "cto") #使用空格作為分隔符,而不是逗號
使用數組
${數組名[索引]}
#引用數組所有元素
${ARRAY_NAME[*]}或${ARRAY_NAME[@]}
刪除數組
#刪除數組中的某一個元素
unset 數組名[索引]
#刪除整個數組
unset 數組名
關聯數組
下標不是預設的數字0..n,而是自定義的。
格式
數組名[自定義的下標]=值
#例如: aa[name1]=bob --- echo ${aa[name1]}--->bob
範例
#隨機生成十個數字,找出最大的和最小的
#!/bin/bash
max=0
min=0
declare -a arr #定義一個數組用於存儲生成的是個隨機數
i=0
while [ $i -le 10 ];do
arr[$i]=$RANDOM #給數組賦值
if [ $i -eq 0 ];then #把第一個數作為基準來判斷後續數的大小
max=${arr[$i]}
min=${arr[$i]}
else
if [ ${arr[$i]} -gt $max ];then
max=${arr[$i]}
fi
if [ ${arr[$i]} -lt $min ];then
min=${arr[$i]}
fi
fi
let i++ #let工具需要一個變數來接受變數的值
done
echo "max is : $max , min is $min"
shell的字元串處理
shell字元串切片
取出某個字元串中的部分內容。
#數組中顯示數組元素的個數:
${#數組名}
#獲取字元串的長度
${#var}
例如:str=jkfjafasfa --->10
#跳過字元串左邊的幾個
${var:num} #num:表示需要跳過幾個
#跳過字元串左邊的幾個,保留多少個
${var:offset:number} #number表示要保留的長度
根據模式取字元串
#刪左留右邊
${var#*word} #從左邊開始找,一直找到word後,就把他們都刪除掉(懶惰模式)
${var##*word} #從左邊開始找,一直找到最右邊的word後,就把他們都刪除掉(貪婪模式)
#範例:
file="var/log/messages" ---echo ${file#*/}-->log/messages
字元串的查找和替換
#替換第一次匹配到的
${var/pattern/substr}
#替換所有匹配到的
${var//pattern/substr}
刪除指定的字元串
#刪除掉第一次找到的
${var/pattern}
#刪除所有找的的
${var/pattern}