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
#description: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
#description: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
#description: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
#description: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
#description: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
#description: 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
#description: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
#description: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
原創文章,作者:a1215276209,如若轉載,請注明出處:http://www.www58058.com/34556