shell 腳本編程 【上】
程序就是指令加上數據組合而成
程序編程風格:
過程式:以指令爲中心,數據服務於指令
象式:以數據爲中心,指令服務於數據
編程語言:
低級:彙編
高級:
編譯:高級語言-->編譯器-->目標代碼 java,C#
解釋:高級語言-->解釋器-->機器代碼 shell, perl, python
shell程序:提供了編程能力,解釋執行
shell腳本是包含一些命令或聲明,並符合一定格式的文 本文件
格式如下:
首行shebang機制
#!/bin/bash
#!/usr/bin/python
#!/usr/bin/perl
那麼shell腳本的用途有哪些呢?
shell腳本可以有:
自動化常用命令
執行系統管理和故障排除
創建簡單的應用程序
處理文本或文件
如何創建腳本
第一步:使用文本編輯器來創建文本文件 第一行必須包括shell聲明序列:#! #!/bin/bash 添加註釋 註釋以#開頭
第二步:運行腳本 給予執行權限,在命令行上指定腳本的絕對或相對路徑 直接運行解釋器,將腳本作爲解釋器程序的參數運行
例如:
#!/bin/bash
#author: wang
#Version: 1.0
#Description:This script displays some information about your
當你寫完一個腳本後你需要檢查 而這時 你可以用到的查錯命令有:
1 bash -n /path/to/some_script
檢測腳本中的語法錯誤
2 bash -x /path/to/some_script
調試執行
變量:命名的內存空間
數據存儲方式: 字符: 數值:整型,浮點型
變量:變量類型
作用: 1、數據存儲格式 2、參與的運算 3、表示的數據範圍
類型: 字符 數值:整型、浮點型
根據變量的生效範圍等標準:
本地變量:生效範圍爲當前shell進程;對當前shell之外 的其它shell進程,包括當前shell的子shell進程均無效 環境變量:生效範圍爲當前shell進程及其子進程 局部變量:生效範圍爲當前shell進程中某代碼片斷(通常 指函數) 位置變量:$1, $2, ...來表示,用於讓腳本在腳本代碼 中調用通過命令行傳遞給它的參數
特殊變量:$?, $0, $*, $@, $#
本地變量
變量賦值:name=‘value’, 可以使用引用value:
(1) 可以是直接字串; name=“root"
(2) 變量引用:name="$USER"
(3) 命令引用:name=`COMMAND`, name=$(COMMAND)
變量引用:${name}, $name
1 "":弱引用,其中的變量引用會被替換爲變量值
2 '':強引用,其中的變量引用不會被替換爲變量值,而保 持原字符串
顯示已定義的所有變量:set
刪除變量:unset name
環境變量
變量聲明、賦值: export name=VALUE declare -x name=VALUE
變量引用:$name, ${name}
顯示所有環境變量: export env printenv
刪除:unset name
bash有許多內建的環境變量:PATH, SHELL, USRE,UID, HISTSIZE, HOME, PWD, OLDPWD, HISTFILE, PS1
只讀和位置變量
只讀變量:只能聲時,但不能修改和刪除 readonly name declare -r name 位置變量:在腳本代碼中調用通過命令行傳遞給腳本的參數
1 $1, $2, ...:對應第1、第2等參數,shift [n]換位置
2 $0: 命令本身
3 $*: 傳遞給腳本的所有參數,全部參數合爲一個字符串
4 $@: 傳遞給腳本的所有參數,每個參數爲獨立字符串
5 $#: 傳遞給腳本的參數的個數
6 $@ $* 只在被雙引號包起來的時候纔會有差異
示例:判斷給出的文件的行數 linecount="$(wc -l $1| cut -d' ' -f1)" echo "$1 has $linecount lines."
shell中的簡單的算術運算和邏輯運算
bash中的算術運算:help let +, -, *, /, %取模(取餘), **(乘方)
實現算術運算:
(1) let var=算術表達式
(2) var=$[算術表達式]
(3) var=$((算術表達式))
(4) var=$(expr arg1 arg2 arg3 ...)
(5) declare –i var = 數值
(6) echo ‘算術表達式’ | bc
乘法符號有些場景中需要轉義,如* bash有內建的隨機數生成器:$RANDOM(1-32767) echo $[$RANDOM%50] :0-49之間隨機數
邏輯運算
true:1
false:0
與: 1 與 1 = 1
1 與 0 = 0
0 與 1 = 0
0 與 0 = 0
或: 1 或 1 = 1
1 或 0 = 1
0 或 1 = 1
0 或 0 = 0
非:!
! 1 = 0
! 0 = 1
短路運算:
短路與: 第一個爲0,結果必定爲0; 第一個爲1,第二個必須要參與運算;
短路或: 第一個爲1,結果必定爲1; 第一個爲0,第二個必須要參與運算;
異或:^ 異或的兩個值,相同爲假,不同爲真
賦值
增強型賦值: +=, -=, *=, /=, %=
let varOPERvalue
例如:let count+=3 自加3後自賦值
自增,自減:
let var+=1
let var++
let var-=1
let var-
聚集命令
有兩種聚集命令的方法:
複合式:date; who | wc -l 命令會一個接一個地運行
子shell:(date; who | wc -l ) >>/tmp/trace 所有的輸出都被髮送給單個STDOUT和STDERR
退出狀態
進程使用退出狀態來報告成功或失敗
0 代表成功,1-255代表失敗
$? 變量保存最近的命令退出狀態
例如: $ ping -c1 -W1 hostdown &> /dev/null $ echo $?
條件測試
判斷某需求是否滿足,需要由測試機制來實現;
專用的測試表達式需要由測試命令輔助完成測試過程;
評估布爾聲明,以便用在條件性執行中 若真,則返回0 若假,則返回1
測試命令: test EXPRESSION [ EXPRESSION ] ` EXPRESSION ` 注意:EXPRESSION前後必須有空白字符
shell中條件性的執行操作符
根據退出狀態而定,命令可以有條件地運行
1 && 代表條件性的AND THEN
2 || 代表條件性的OR ELSE
例如: $ grep -q no_such_user /etc/passwd \ || echo 'No such user' No such user $
ping -c1 -W2 station1 &> /dev/null \ > && echo "station1 is up" \ > || (echo 'station1 is unreachable'; exit 1) station1 is up
test命令
長格式的例子: $ test "$A" == "$B" && echo "Strings are equal" $ test “$A” -eq “$B” \ && echo "Integers are equal"
簡寫格式的例子: $ [ "$A" == "$B" ] && echo "Strings are equal" $ [ "$A" -eq "$B" ] && echo "Integers are equal"
bash的測試類型 有:
1 數值測試
2 字符串測試
3 存在性測試
4 文件權限測試
5 文件大小測試
6 雙目測試
數值測試:
-gt: 是否大於;
-ge: 是否大於等於;
-eq: 是否等於;
-ne: 是否不等於;
-lt: 是否小於;
-le: 是否小於等於;
字符串測試:
==:是否等於;
>: ascii碼是否大於ascii碼
<: 是否小於
!=: 是否不等於
=~: 左側字符串是否能夠被右側的PATTERN所匹配
注意: 此表達式一般用於[[ ]]中;
-z "STRING":字符串是否爲空,空爲真,不空爲假
-n "STRING":字符串是否不空,不空爲真,空爲假
注意:用於字符串比較時的用到的操作數都應該使用引號
存在性測試
-a FILE:同-e
-e FILE: 文件存在性測試,存在爲真,否則爲假;
存在性及類別測試
-b FILE:是否存在且爲塊設備文件;
-c FILE:是否存在且爲字符設備文件;
-d FILE:是否存在且爲目錄文件;
-f FILE:是否存在且爲普通文件;
-h FILE 或 -L FILE:存在且爲符號鏈接文件;
-p FILE:是否存在且爲命名管道文件;
-S FILE:是否存在且爲套接字文件;
文件權限測試:
-r FILE:是否存在且可讀
-w FILE: 是否存在且可寫
-x FILE: 是否存在且可執行
文件特殊權限測試:
-g FILE:是否存在且擁有sgid權限;
-u FILE:是否存在且擁有suid權限;
-k FILE:是否存在且擁有sticky權限
文件大小測試:
-s FILE: 是否存在且非空; 文件是否打開:
-t fd: fd表示文件描述符是否已經打開且與某終端相關 -N FILE:文件自動上一次被讀取之後是否被修改過
-O FILE:當前有效用戶是否爲文件屬主
-G FILE:當前有效用戶是否爲文件屬組
雙目測試:
FILE1 -ef FILE2: FILE1與FILE2是否指向同一個設 備上的相同inode
FILE1 -nt FILE2: FILE1是否新於FILE2;
FILE1 -ot FILE2: FILE1是否舊於FILE2;
組合測試條件
第一種方式: COMMAND1 && COMMAND2 並且 COMMAND1 || COMMAND2 或者 ! COMMAND 非 如:[ -e FILE ] && [ -r FILE ]
第二種方式: EXPRESSION1 -a EXPRESSION2 並且 EXPRESSION1 -o EXPRESSION2 或者 ! EXPRESSION
必須使用測試命令進行; # [ -z “$HOSTNAME” -o $HOSTNAME "==\ "localhost.localdomain" ] && hostname www.magedu.com # [ -f /bin/cat -a -x /bin/cat ] && cat /etc/fstab
用read命令的使用
使用read來把輸入值分配給一個或多個shell變量:
-p 指定要顯示的提示
-t TIMEOUT read 從標準輸入中讀取值,給每個單詞分配一個變量 所有剩餘單詞都被分配給最後一個變量
read -p “Enter a filename: “ FILE
練習題
1、編寫腳本/root/bin/systeminfo.sh,顯示當前主機系統信息,包括主機名,IPv4地址,操作系統版本,內核版本,CPU型號,內存大小,硬盤大小。
#!/bin/bash
#descrition:show some information about hostname ipv4address version kernel cpu disk and memory
Hostname=`hostname`
Ipv4=`ifconfig | sed -n '2p'|sed -r 's#.*addr:(.*) .*B.*#\1#'`
version=`cat /etc/redhat-release`
kernel=`uname -r`
Cpu=`lscpu | sed -n '/^Model name.*/p'|sed -r 's@.*[[:space:]]{3}+(.*$)@\1@'`
memory=`free -h |tr -s " "|cut -d " " -f2 | sed -n '2p'`
disk=`fdisk -l |sed -n '2p'| sed -r 's@.* (.*) GB.*@\1@'`
echo 'hostname :' $Hostname
echo 'IPv4:' $Ipv4
echo 'OS version :' $version
echo 'Kernel :' $kernel
echo 'CPU :' $Cpu
echo 'memory:' $memory
echo "disk: $disk"
2、編寫腳本/root/bin/backup.sh,可實現每日將/etc/目錄備份到/root/etcYYYY-mm-dd中
#!/bin/bash
#descrition:copy
echo 'copy /etc ....'
cp -a /etc /root/${date +%F}
echo 'copy is over'
[root@localhost bin]# bash backup.sh
copy /etc ....
copy is over
3、編寫腳本/root/bin/disk.sh,顯示當前硬盤分區中空間利用率最大的值、
#!/bin/bash
#show the using max rate about disk
max=`df |tr -s ' ' '%'|cut -d '%' -f5|sort -n |tail -1`
echo "maxrate:$max"
[root@localhost bin]# bash disk.sh
maxrate:47
4、編寫腳本/root/bin/links.sh,顯示正連接本主機的每個遠程主機的IPv4地址和連接數,並按連接數從大到小排序
#!/bin/bash
#show the links
links=`netstat -nt |tr -s ' ' |tail -n +3 | cut -d " " -f5 | sed -r 's@(.*):.*@\1@'|sort |uniq -c`
echo "$links"
[root@localhost bin]# bash link.sh
1 10.1.25.29
5、寫一個腳本/root/bin/sumid.sh,計算/etc/passwd文件中的第10個用戶和第20用戶的ID之和
#!/bin/bash
#descrition:sumid
sumid=`sed -n '10p;20p' /etc/passwd|cut -d : -f 3 |tr '\n' '+' | sed -r 's#(.*)\+#\1\n#'| bc`
echo "sumid :$sumid"
[root@localhost bin]# bash sumid.sh
sumid :80
6、寫一個腳本/root/bin/sumspace.sh,傳遞兩個文件路徑作爲參數給腳本,計算這兩個文件中所有空白行之和
##!/bin/bash
#descrition:sum about space
space1=`grep '^$' $1 | wc -l`
space2=`grep '^$' $2 | wc -l`
Sumspace=$space1+$space2
echo "sumspace: $Sumspace"
[root@localhost bin]# bash sumspace.sh /etc/rc.d/init.d/functions ../.bash_profile
sumspace: 109
7、寫一個腳本/root/bin/sumfile.sh,統計/etc, /var, /usr目錄中共有多少個一級子目錄和文件
#!/bin/bash
#descrition:sum about directory
etcnum=`ls -A -1 /etc/ |wc -l`
varnum=`ls -A -1 /var/ |wc -l`
usernum=`ls -A -1 /usr/ |wc -l`
sum=$etcnum+$varnum+$usernum
sum:$sum"
[root@localhost bin]# bash sumfile.sh
sum:299
8、寫一個腳本/root/bin/argsnum.sh,接受一個文件路徑作爲參數;如果參數個數小於1,則提示用戶“至少應該給一個參數”,並立即退出;如果參數個數不小於1,則顯示第一個參數所指向的文件中的空白行數
#!/bin/bash
#
[[ $# -lt 1 ]] && echo "至少輸入一個參數" || (grep '^$' $1 | wc -l)
[root@localhost bin]# bash argsnum.sh /etc/rc.d/init.d/functions
105
[root@localhost bin]# bash argsnum.sh
至少輸入一個參數
9、寫一個腳本/root/bin/hostping.sh,接受一個主機的IPv4地址做爲參數,測試是否可連通。如果能ping通,則提示用戶“該IP地址可訪問”;如果不可ping通,則提示用戶“該IP地址不可訪問”
#!/bin/bash
#descrition: how to ping
ping -w1 -c1 "$1" &>/dev/null
Ping=`echo $?`
[[ $Ping -eq 0 ]] && (echo "該IP地址可訪問") || (echo "該IP地址不可訪問")
[root@localhost bin]# bash hostping.sh 10.1.25.29
該IP地址可訪問
[root@localhost bin]# bash hostping.sh 10.1.25.155
該IP地址不可訪問
10、chmod -rw /tmp/file1,編寫腳本/root/bin/per.sh,判斷當前用戶對/tmp/fiile1文件是否不可讀且不可寫
#!/bin/bash
#
(([ ! -r /tmp/flie1 ]) && ([ ! -w /tmp/file1 ])) && (echo "此用戶對/tmp/file1文件不可讀寫" )
11、編寫腳本/root/bin/nologin.sh和login.sh,實現禁止和充許普通用戶登錄系統。
#!/bin/bash
#descrition:nologin and login
[ -f /etc/nologin ] && (echo "normal user can not login")|| (touch /etc/nologin)
12、
寫一個腳本/root/bin/hostping.sh,接受一個主機的IPv4地址做爲參數,先判斷是否合格IP,否,提示IP格式不合法並退出,是,
測試是否可連通。如果能ping通,則提示用戶“該IP地址可訪問”;如果不可ping通,則提示用戶“該IP地址不可訪問”
#!/bin/bash
#descrition:check ip
echo $1 | egrep -o '([0-9]|([1-9][0-9])|(1[0-9]{2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1[0-9]{2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1[0-9]{2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1[0-9]{2})|(2[0-4][0-9])|(25[0-5]))\>'&>/dev/null
result=`echo $?`
[ result -ne 0 ] && echo "it not ip" && exit 222
ping -w1 -c1 "$1" &>/dev/null && (echo "you can access this ip") || (echo "you can not access this ip")
13、計算1+2+3+…+100的值
[root@qzx ~]# seq -s + 1 100 |bc
5050
14、計算從腳本第一參數A開始,到第二個參數B的所有數字的總和,判斷B是否大於A,否提示錯誤並退出,是則計算之
#!/bin/bash
#計算從腳本第一參數A開始,到第二個參數B的所有數字的總和,判斷B是否大於A,否提示錯誤並退出,是則計算之
[ $2 -gt $1 ] && (seq -s + $1 $2 | egrep -o '.*[^+]' |bc) || (echo "the number is wrong")
[root@localhost bin]# bash numA_B.sh 1 100
5050