Shell基礎篇:基本語法

一 什麼是shell>

  將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 ]
then
echo "$var is directory"
elif [ -b $var ]
then
echo "$var is block"
elif [ -f $var ]
then
echo "$var is regular file"
else
echo 'unknow'
fi

if 測試中還可以執行命令 根據命令的返回值做判斷
# if cd / ;then echo Y ;fi
# if grep -q root /etc/passwd ;then echo Y ;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 ]
then
echo "$1 is directory"
elif [ -b $1 ]
then
echo "$1 is block"
elif [ -f $1 ]
then
echo "$1 is regular file"
else
echo '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 :
do
read -p 'please input your name: ' name
read -p 'please input your password: ' pwd
if [ $name = 'egon' ] && [ $pwd = '123' ]
then
echo 'login sucessful'
break #continue
fi
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))
do
echo $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 :
do
clear
echo -e "A:${var1}\\nB:${var2}\\nC:${var3}"
temp=$var1
var1=$var2
var2=$var3
var3=$temp
sleep 1
done

2.wihle和read實現逐行處理

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

[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/bash

for i in {1..254}
do
(ping -W 1 -c 1 192.168.1.$i &> /dev/null && echo 192.168.1.$i) &
done


讓文件測試腳本支持多個參數
#!/bin/bash

for i in $@
do
if [[ -d $i ]]
then
echo "$i is directory."
elif [[ -b $i ]]
then
echo "$i is block device."
elif [[ -f $i ]]
then
echo "$i is a regular file."
else
echo "unknow."
fi
done

多個for嵌套
嵌套for中使用

continue:默認退出本次循環

break:默認退出本層循環

可以直接在命令行寫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

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
do
case $var in
p|P|cpu|CPU)
echo -e "\\n\\n"
grep 'model name\\|cpu MHz\\|processor' /proc/cpuinfo |sort |uniq
echo -e "\\n\\n"

;;
m|m|mem|MEM)
echo -e "\\n\\n"
free
echo -e "\\n\\n"
;;
d|D|disk|DISK)
echo -e "\\n\\n"
df -Th
echo -e "\\n\\n"
;;
h|H|help|HELP)
echo -e "\\n\\tcommand\\taction\\n\\n"
for i in cpu mem disk
do
echo -e "\\t$i\\t${i}_info"
done
echo -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 選看內容

瞭解: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,其它值為失敗)


分享到:


相關文章: