shell 腳本編程 【上】

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


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章