日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

Shell第三篇:基本语法

發布時間:2025/4/9 编程问答 20 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Shell第三篇:基本语法 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

一 什么是shell script

  將OS命令堆積到可執行的文件里,由上至下的順序執行文本里的OS命令 就是腳本了.
  再加上些智能(條件/流控)控制,就變成了智能化腳本了.

二 變量

part1 為何要有變量

程序的運行就是一些列狀態的變量->用變量值的變化去表示

part2 變量命名規則

以字母或下劃線開頭,剩下的部分可以是:字母、數字、下劃線.

最好遵循下述規范:

1.以字母開頭
2.使用中劃線或者下劃線做單詞的連接
3.同類型的用數字區分
4.對于文件最好加上拓展名
例如: sql_bak.tar.gz,log_bak.tar.bz2?

part3 系統變量

set 和 env區別
set:顯示所有變量
env:環境變量

part4 變量賦值

VARNAME=VALUE
echo $VARNAME
刪除變量 unset VARNAME

part5 常用系統變量

PATH
PWD
LANG
HOME
HISTSIZE
PS1
IFS
域分隔符 是空格,換行,TAB鍵的合集

part6 全局變量與局部變量

[root@MiWiFi-R3-srv ~]# gender='male' #在爹這個位置定義一個局部變量gender
[root@MiWiFi-R3-srv ~]# export money=1000 #在爹這個位置定義一個全局變量money
[root@MiWiFi-R3-srv ~]#
[root@MiWiFi-R3-srv ~]#
[root@MiWiFi-R3-srv ~]# bash #切換到子bash
[root@MiWiFi-R3-srv ~]# echo $gender #在兒子這里看它爹的局部變量gender,結果為空->看不到

[root@MiWiFi-R3-srv ~]# echo $money #在兒子這里看它爹的全局變量money,可以看到
1000
[root@MiWiFi-R3-srv ~]#
[root@MiWiFi-R3-srv ~]# export hobby='piao' #在兒子這里定義一個全局變量hobby
[root@MiWiFi-R3-srv ~]# exit #退出,進入爹的bash環境
exit
[root@MiWiFi-R3-srv ~]# echo $hobby #爹是看不到兒子的export的,兒子的兒子可以看到

[root@MiWiFi-R3-srv ~]#

part6 定義變量名的邊界

[root@MiWiFi-R3-srv ~]# rest_mem=20
[root@MiWiFi-R3-srv ~]# echo ${rest_mem}%
20%

part 7 數據類型

bash中的變量無須聲明,拿來就用.默認的變量都會是字符類型,還可以有數字類型,普通的腳本,這兩種類型夠用了

三 運算符

part1 算術運算符

+

-

*

/

%

[root@MiWiFi-R3-srv ~]# echo $[3+1]
4

part2 關系操作

與(())連用

<

>

<=

>=

==

!=

&&

||

test命令相關,[]可以達到一樣的效果
[root@MiWiFi-R3-srv ~]# x=1
[root@MiWiFi-R3-srv ~]# [ $x -gt 1 ]
[root@MiWiFi-R3-srv ~]# echo $?
0

part3 賦值運算符

=

+=

*=

/=

%=

[root@MiWiFi-R3-srv ~]# x=10
[root@MiWiFi-R3-srv ~]# ((x%3))
[root@MiWiFi-R3-srv ~]# echo $x
10
[root@MiWiFi-R3-srv ~]#
[root@MiWiFi-R3-srv ~]# ((x%=3))
[root@MiWiFi-R3-srv ~]# echo $x
1

part4 shell里的所有計算器
$[] (()) $(()) expr bc bc -l

浮點運算:yum install bc -y

[root@MiWiFi-R3-srv ~]# echo 'scale=2;1/3'|bc -l
.33

part5 測試操作

命令執行后會返回到一個系統變量中 $?
如果$?值為0 表示命令執行成功 否則為失敗


測試命令 test [ ] [[ ]] (( ))
打開man test 逐一介紹每個參數
part5-1、測試文件狀態
-d 目錄
-s 文件長度 > 0、非空
-f 正規文件
-w 可寫

-r 可讀

-x 可執行

-L 符號連接

-u 文件有 suid 位設置

part5-2、字符串測試
= 兩個字符串相等
!= 兩個字符串不相等
-z 空串
-n 非空串

[root@MiWiFi-R3-srv ~]# var1='abc'
[root@MiWiFi-R3-srv ~]# var2='123'
[root@MiWiFi-R3-srv ~]# [ $var1 == $var2 ]
[root@MiWiFi-R3-srv ~]# echo $?
1

part5-3、測試數值
-eq 等于
-ne 不等于
-gt 大于
-lt 小于
-ge 大于等于
-le 小于等于

[root@MiWiFi-R3-srv ~]# [ 10000 -gt 250 ] #不要使用大于號小于號等于號等,要使用man test中規定的,詳見下一小節4拓展
[root@MiWiFi-R3-srv ~]# echo $?
0

part5-4、拓展測試符號 [[ ]] (())
數字測試符號
# [ 10 < 2 ] # 語法錯誤
-bash: 2: 沒有那個文件或目錄
#

# [[ 2 > 10 ]] # 結果錯誤
# echo $?
0
# [[ 20 > 10 ]] # 正確
# echo $?
0
# (( 10 < 20 ))
# echo $?
0

字符測試
# [ "aa" = "aa" ]
# echo $?
0
# [[ "aa" = "aa" ]]
# echo $?
0
# (( "aa" = "aa" )) #結果錯誤
# echo $?
1
混合測試
# [ a = a -a 10 < 20 ]
-bash: 20: 沒有那個文件或目錄
[root@seker ~]# [[ a = a -a 10 < 20 ]]
-bash: syntax error in conditional expression
-bash: syntax error near `-a'
[root@seker ~]# [[ a = a && 10 < 20 ]]
[root@seker ~]# echo $?
0
[root@seker ~]# [[ a = a || 10 < 20 ]]
[root@seker ~]# echo $?
0
[root@seker ~]# (( a = a || 10 < 20 ))
[root@seker ~]# echo $?
0
[root@seker ~]# (( a = a && 10 < 20 ))
[root@seker ~]# echo $?
0
[root@seker ~]#
結論:
比較數字,使用(( ))
其他測試使用 [[ ]]
包含數字比較的混合測試,使用[[ expr1 && expr2 ]] (( expr1 || expr2 ))

兩個文件的比較
FILE1 -ef FILE2
測試兩個文件是否是相同的inode
有時為了找到同一個INODE號的文件 更傾向于使用 find 命令的 -inum 或 --samefile

FILE1 -nt FILE2
FILE1 is newer (modification date) than FILE2

FILE1 -ot FILE2
FILE1 is older than FILE2

四 流程控制

part1分支結構

#!/bin/bash var='/etc/init.d' #var='/dev/sda' if [ -d $var ]thenecho "$var is directory" elif [ -b $var ]thenecho "$var is block" elif [ -f $var ]thenecho "$var is regular file" elseecho 'unknow' fi

?

if 測試中還可以執行命令 根據命令的返回值做判斷
# if cd / ;then echo Y ;fi
# if grep -q root /etc/passwd ;then echo Y ;fi

#!/bin/bash username='egon' password='123' read -p 'user: ' name read -p 'passwd: ' passwdif [ $name = $username -a $passwd = $password ];thenecho 'login successful' elseecho 'username or password err' fi 用戶驗證 #!/bin/bash age=87 read -p 'num: ' nif [ $n -eq $age ];thenecho 'you get it' elif [ $n -gt $age ];thenecho 'too big' elif [ $n -lt $age ];thenecho 'too small' fi 猜老男孩的年紀 #!/bin/bash read -p 'your score: ' scoreif [ $score -ge 90 ];thenecho '優秀' elif [ $score -ge 70 -a $score -lt 90 ];thenecho '良好' elif [ $score -ge 60 -a $score -lt 70 ];thenecho '一般' elif [ $score -lt 60 ];thenecho '較差' fi 查詢成績

?

?

向腳本傳遞參數

#test.sh echo $0 echo $1 echo $2 echo $3 echo ${11} echo '$$' $$ echo '$*' $* echo '$@' $@ echo '$#' $# echo '$?' $?''' 測試:python test.sh 1 2 3 4 5 6 7 8 9 10 11 輸出結果: ./test.sh 1 2 3 11 $$ 14312 $* 1 2 3 4 5 6 7 8 9 10 11 $@ 1 2 3 4 5 6 7 8 9 10 11 $# 11 $? 0 '''

?

修改腳本,使其能接收調用者傳來的參數

[root@MiWiFi-R3-srv ~]# cat test_file.sh #!/bin/bash if [ -d $1 ]thenecho "$1 is directory" elif [ -b $1 ]thenecho "$1 is block" elif [ -f $1 ]thenecho "$1 is regular file" elseecho 'unknown' fi [root@MiWiFi-R3-srv ~]# ./test_file.sh /etc/passwd /etc/passwd is regular file

?

part2 循環結構

?

part2-1 while循環

while (條件)

do
動作
done

需要無限循環時我們會選擇while :

[root@MiWiFi-R3-srv ~]# cat login.sh #!/bin/bashwhile : doread -p 'please input your name: ' nameread -p 'please input your password: ' pwdif [ $name = 'egon' ] && [ $pwd = '123' ]thenecho 'login sucessful'break #continuefi done [root@MiWiFi-R3-srv ~]# ./login.sh please input your name: egon please input your password: 123 login sucessful

?

[root@MiWiFi-R3-srv ~]# cat 1.sh #!/bin/bash i=1 while ((i<10)) doecho $i((i++)) done [root@MiWiFi-R3-srv ~]# ./1.sh 1 2 3 4 5 6 7 8 9

?

?

練習:
1.while死循環

[root@MiWiFi-R3-srv ~]# cat 1.sh #!/bin/bash var1=AAA var2=BBB var3=CCC while : doclearecho -e "A:${var1}\nB:${var2}\nC:${var3}"temp=$var1var1=$var2var2=$var3var3=$tempsleep 1 done

?

2.wihle和read實現逐行處理

[root@MiWiFi-R3-srv ~]# cat 1.sh #!/bin/bash while read var doecho $((++i)):$var done</etc/passwd

[root@MiWiFi
-R3-srv ~]# ./1.sh 1:root:x:0:0:root:/root:/bin/bash 2:bin:x:1:1:bin:/bin:/sbin/nologin 3:daemon:x:2:2:daemon:/sbin:/sbin/nologin 4:adm:x:3:4:adm:/var/adm:/sbin/nologin ...........

?


part2-2 for循環

C語言格式的for

for ((i=1;i<=10;i++)) do echo $i done

?


shell格式的for

for i in {1..10} do echo $i done

?

?

shell的for,常用in列表方式

for i in 1 2 3 for i in {1,2,3} for i in {1..9} for i in {9..1} for i in {a..z} for i in {A..Z} for i in {X..Z} for i in $(cmd) for i in $(find ...)

?

?

小例子

檢查內網存活的IP#!/bin/bashfor i in {1..254}do(ping -W 1 -c 1 192.168.1.$i &> /dev/null && echo 192.168.1.$i) &done讓文件測試腳本支持多個參數#!/bin/bashfor i in $@doif [[ -d $i ]]thenecho "$i is directory."elif [[ -b $i ]]thenecho "$i is block device."elif [[ -f $i ]]thenecho "$i is a regular file."elseecho "unknow."fidone

?

?

多個for嵌套
嵌套for中使用

continue:默認退出本次循環

break:默認退出本層循環

使用break: break 默認參數是 1 所以寫 break 等于 break 1 意義:退出當前循環層 break 2 則向上退出2層循環 當前循環也計算在退出層次里for i in {1..9} do for j in {0..9} do for n in {0..9} do echo $i$j$n if ((n==5)) then break 3 fi done done done使用continue continue = continue 1 在當次循環中忽略continue后續的代碼 就是:立即結束當前循環中的當次循環,而轉入當前循環的下一次循環continue 2 = break 1 continue 3 = break 2 .... 依次類推for i in {1..9} do for j in {0..9} do for n in {0..9} do echo $i$j$n if ((n==5)) then continue echo "-----------------------------" fi done done done 了解即可

?

?

可以直接在命令行寫for循環

# for i in {1..10};do [ $i -eq 5 ] && continue || echo $i;done
# for i in {1..10};do [ $i -eq 5 ] && break || echo $i;done

?


練習:
統計/dev下每種類型文件的數量

#!/bin/bash dir='/dev' for i in `ls $dir` doif [ -b $dir/$i ]then((block++))elif [ -f $dir/$i ]then((file++))elif [ -d $dir/$i ]then((directory++))else((unkown++))fi doneecho 'block' $block echo 'regular file' $file echo 'directory' $directory echo 'unkown' $unkown View Code

?

向腳本傳遞一個用戶名,驗證這個用戶是否存在.

[root@MiWiFi-R3-srv ~]# cat testuser.sh #!/bin/bash id $1 &> /dev/null if [ $? -eq 0 ];thenecho "用戶$1存在" elseecho "用戶$1不存在" fi [root@MiWiFi-R3-srv ~]# ./testuser.sh root 用戶root存在 View Code

添加30個用戶,再將它們刪除

for i in {1..30}; douseradd user$i&&echo "user$i create successful" donefor i in {1..30}; douserdel -r user$i&&echo "user$i delete successful" done View Code

?

part2-3?case語句

read -p "username: " -t 5 uname
echo
if [[ -z $uname ]]
then
uname=default
fi

case $uname in
root)
echo "welcome $uname"
;;
seker)
echo "welcome $uname"
;;
default)
echo "welcome $uname"
;;
*)
echo "no user $uname"
esac

part2-4 綜合到一起,制作一個簡單的菜單功能

#!/bin/bash echo "script name: `basename $0`" echo "version 1.0" echo "date 2017-03-23" echo "Author: biubiu" while read -p "(h for help): " var docase $var in p|P|cpu|CPU)echo -e "\n\n"grep 'model name\|cpu MHz\|processor' /proc/cpuinfo |sort |uniqecho -e "\n\n";;m|m|mem|MEM)echo -e "\n\n"freeecho -e "\n\n";;d|D|disk|DISK)echo -e "\n\n"df -Thecho -e "\n\n";;h|H|help|HELP)echo -e "\n\tcommand\taction\n\n"for i in cpu mem diskdoecho -e "\t$i\t${i}_info"doneecho -e "\thelp\tthis help page.."echo -e "\tquit\texit !!.."echo -e "\n\n";;q|Q|quit|exit)exit;;*)echo -e "\n$var Enter Error...\n"esac done

?

part2-5 選看內容

# cat select.sh #!/bin/bash PS3='choose one: ' #select 默認使用PS3做提示符 echo select var in $(for i in {A..D};do echo $i;done) do echo echo "your choose is $var" echo "OK" echo break # 跳出select,否則是死循環 done # # ./select.sh1) A 2) B 3) C 4) D choose one: 3your choose is C OK# 若省略 in list 則把 $@ 做列表項 # cat select.sh #!/bin/bash PS3='choose one: ' echo select var do echo echo "your choose is $var" echo "OK" echo break done # # ./select.sh A B C D1) A 2) B 3) C 4) D choose one: Cyour choose is OK 了解:select 菜單

五 函數

交互shell中的函數
function abc(){ echo 'aaa';echo 'bbbb'; }

set 查看

調用 abc
[root@seker ~]# abc
aaa
bbbb
[root@seker ~]#


腳本中的函數

1.函數定義
shell允許將一組命令集或語句形成一個可用塊,這些塊稱為shell函數
定義函數的格式:

function-name (){
command1
........
}
或 function function-name(){ #函數名前面多了個function關鍵字
command1
........
}
2.函數調用
以下是一個函數的腳本實例:
#!/bin/bash
#hello
function hello(){ #聲明函數
echo "Hello!" #函數的主體,輸出"Hello!"
} #函數結束
hello #調用函數
3.參數傳遞
向函數傳遞參數就像在腳本是使用變量位置$1,$2,$3...$9
以下是一個傳遞參數的實例:
#!/bin/bash
#hellofun
function hello(){
echo "Hello! The first parameter is '$1'."
}
hello good
#該腳本執行的結果是: Hello! The first parameter is 'good'.
4.函數文件
保存函數的文件,用以上的例子寫成一個函數文件如下:
#!/bin/bash
#hellofunction
function hello(){
echo "Hello!"
return 1
}
上面的hellofunction文件就是一個函數文件,可通過另一個腳本來調用
#!/bin/bash
#hellof
. hellofunction #注意點和hellofunction之間有個空格
hello
5.載入和刪除
用set查看已載入的函數
用unset function-name 取消載入
舉例如下:
#!/bin/bash
#hellof
. hellofunction
unset hello
hello #因為已經取消載入。。所以會出錯
6.函數返回狀態
#!/bin/bash
#hellofun
function hello(){
echo "Hello! The first parameter is '$1'."
return 1
}
hello
echo $? #輸出返回的狀態值(一般成功是返回0,其它值為失敗)

六 計劃任務crontab

http://www.cnblogs.com/linhaifeng/articles/6045600.html#_label21

七 補充:發送郵件

發件箱:python4_mail@163.com alex3714 ?自定義客戶端登錄帳號和密碼分別為:python4_mail@163.com sbalex3714

收件箱:python4_recvmail@163.com alex371

?

python通過SMTP發送郵件失敗:
錯誤1:smtplib.SMTPAuthenticationError: (550, b‘User has no permission‘)
?? ?我們使用python發送郵件時相當于自定義客戶端根據用戶名和密碼登錄,然后使用SMTP服務發送郵件,新注冊的163郵箱是默認不開啟客戶端授權的(對指定的郵箱大師客戶端默認開啟),因此登錄總是被拒絕,解決辦法(以163郵箱為例):進入163郵箱-設置-客戶端授權密碼-開啟(授權碼是用于登錄第三方郵件客戶端的專用密碼)
錯誤2:smtplib.SMTPAuthenticationError: (535, b‘Error: authentication failed‘)
  以163郵箱為例,在開啟POP3/SMTP服務,并開啟客戶端授權密碼時會設置授權碼,將這個授權碼代替smtplib.SMTP().login(user,password)方法中的password即可。

#!/usr/bin/python # -*- coding: UTF-8 -*- import smtplib import email.mime.multipart import email.mime.textmsg = email.mime.multipart.MIMEMultipart()msg['Subject'] = '你是風兒我是沙,纏纏綿綿回我家' msg['From'] = 'python4_mail@163.com' msg['To'] = 'python4_recvmail@163.com' content = '''來來來,一起搖擺''' txt = email.mime.text.MIMEText(content,_charset='utf-8') msg.attach(txt)smtp = smtplib.SMTP() smtp.connect('smtp.163.com', '25') smtp.login('python4_mail', 'sbalex3714') smtp.sendmail('python4_mail@163.com', 'python4_recvmail@163.com', msg.as_string()) smtp.quit() print('郵件發送成功email has send out !') python郵件發送工具(smtplib)

?

?

練習1:監控主機的cpu,內存,磁盤,超過閾值則發送報警郵件

步驟一:準備發送郵件的工具

#!/usr/bin/python # -*- coding: UTF-8 -*- import sys import smtplib import email.mime.multipart import email.mime.textserver = 'smtp.163.com' port = '25'def sendmail(server,port,user,pwd,msg):smtp = smtplib.SMTP()smtp.connect(server,port)smtp.login(user, pwd)smtp.sendmail(msg['from'], msg['to'], msg.as_string())smtp.quit()print('郵件發送成功email has send out !')if __name__ == '__main__':msg = email.mime.multipart.MIMEMultipart()msg['Subject'] = '你是風兒我是沙,纏纏綿綿回我家'msg['From'] = 'python4_mail@163.com'msg['To'] = 'python4_recvmail@163.com'user = 'python4_mail'pwd = 'sbalex3714'content='%s\n%s' %('\n'.join(sys.argv[1:4]),' '.join(sys.argv[4:])) #格式處理,專門針對我們的郵件格式 txt = email.mime.text.MIMEText(content, _charset='utf-8')msg.attach(txt)sendmail(server,port,user,pwd,msg)

步驟二:將上述文件內容拷貝到/usr/bin/mail并chmod+x /usr/bin/mail

步驟三:然后新建監控腳本servermonitor.sh

#!/bin/bash cpu_limit=0 #cpu使用超過90%則報警,此處我們為了得到報警郵件的實驗效果,直接設置成0 mem_limit=0 #內存使用超過90%則報警,同上 disk='/dev/sda1' #需要監控的磁盤名 disk_inode_limit=0 #磁盤inode使用超過90%則報警,同上 disk_space_limit=0 #磁盤空間使用超過90%則報警,同上 function monitor_cpu(){cpu_free=`vmstat 1 5 |awk 'NR>=3{x = x + $15} END {print x/5}' |awk -F. '{print $1}'`cpu_use=$((100-cpu_free))if [ $cpu_use -gt $cpu_limit ]thenmsg="TIME:$(date +%F_%T) HOSTNAME:$(hostname)IPADDR:$(ifconfig |awk 'NR==2{print $2}')MSG:CPU usage exceeds the limit,current value is ${cpu_use}%" echo $msg/usr/bin/mail $msgfi }function monitor_mem(){mem_total=`free |awk 'NR==2{print $2}'`mem_use=`free |awk 'NR==2{print $3}'`mem_per=`echo "scale=2;$mem_use/$mem_total" |bc -l|cut -d. -f2`if [ $mem_per -gt $mem_limit ]thenmsg="TIME:$(date +%F_%T) HOSTNAME:$(hostname)IPADDR:$(ifconfig |awk 'NR==2{print $2}')MSG:Memory usage exceeds the limit,current value is ${mem_per}%" echo $msg/usr/bin/mail $msgfi }function monitor_disk_inode(){inode_use=`df -i $disk |awk 'NR==2{print $5}' |cut -d% -f1`if [ $inode_use -gt $disk_inode_limit ]thenmsg="TIME:$(date +%F_%T) HOSTNAME:$(hostname)IPADDR:$(ifconfig |awk 'NR==2{print $2}')MSG:Disk inode usage exceeds the limit,current value is ${inode_use}%" echo $msg/usr/bin/mail $msgfi }function monitor_disk_space(){space_use=`df $disk |awk 'NR==2{print $5}'|cut -d% -f1`if [ $space_use -gt $disk_space_limit ]thenmsg="TIME:$(date +%F_%T) HOSTNAME:$(hostname)IPADDR:$(ifconfig |awk 'NR==2{print $2}')MSG:Disk space usage exceeds the limit,current value is ${space_use}%" echo $msg/usr/bin/mail $msgfi }monitor_cpu &>> /tmp/monitor.log monitor_mem &>> /tmp/monitor.log monitor_disk_inode &>> /tmp/monitor.log monitor_disk_space &>> /tmp/monitor.log

?

步驟四:編寫計劃任務

?

* * * * * /root/servermonitor.sh

結果:收到的報警郵件形式為(發送郵件會受主機名解析,163郵箱自動屏蔽等方面的影響,因而測試時最好是基于自己搭建的郵箱)

?

?

ps:服務器收到的報警郵件如下(zabbix監控)

?

?

?

練習2:編寫自動部署腳本?

安裝腳本的寫法就很簡單了,難點應該是在修改配置文件

[root@www tmp]# msg='upstream { server 1.1.1.1;server 2.2.2.2;server 3.3.3.3}'
[root@www tmp]# sed -ri "/^http/a $msg" nginx.conf #增加upstream

?

?

[root@www tmp]# sed -ri "/^ *location \/ \{$/a proxy_pass http://my_upstream\;" nginx.conf ?#修改localtion

?

轉載于:https://www.cnblogs.com/linhaifeng/p/6602149.html

總結

以上是生活随笔為你收集整理的Shell第三篇:基本语法的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。