從頭一二去閱讀語法和命令說明,對於腳本小白來說比較枯燥,難以堅持,所以這裡選擇對一份完整的shell腳本代碼來逐行逐段解讀,希望可以一渡小白,幫助我們快速進入腳本的大門^_^ ...
以下內容為本人的學習筆記,如需要轉載,請聲明原文鏈接 微信公眾號「englyf」https://www.cnblogs.com/englyf/p/16721350.html
從頭一二去閱讀語法和命令說明,對於腳本小白來說比較枯燥,難以堅持,所以這裡選擇對一份完整的shell腳本代碼來逐行逐段解讀,希望可以一渡小白,幫助我們快速進入腳本的大門_。
司機要開車了:
#!/bin/sh
用註釋的形式說明文件打開類型,此處意指本腳本需要用 /bin/sh 打開。
#V1.0 Added hardware type detection
#V1.1 xxxx-xx-xx 重構腳本
#V1.2 xxxx-xx-xx 將日誌信息輸出到U盤
#V1.3 xxxx-xx-xx 讀取存儲的檢測結果值
#V1.4 xxxx-xx-xx 添加校驗程式版本,U盤下有對應版本信息文件才校驗
上面幾行表示註釋內容。單行註釋,以#
開始
UDiskMountDir=$(df | grep /mnt/sd | awk 'END{print $NF}')
聲明變數 UDiskMountDir 並賦值。
$()
括弧內填入命令並執行,最後返回輸出到變數 UDiskMountDir。
df
命令顯示liinux系統上的文件系統磁碟使用情況統計。後邊還可以帶選項,執行更複雜的輸出。命令後邊加 '|' 表示將此命令的輸出作為後邊緊接著的命令的輸入。
grep
命令用於對文本按行搜索然後輸出該行。英文全稱 global search regular expression(RE) and print out the line。grep /mnt/sd
表示搜索輸入信息里包含 /mnt/sd
的一行,並輸出該行。
awk
命令用於對文本處理,END
表示執行最後的運算或者列印最終的輸出結果,這裡用於列印輸出,$NF
表示列數,print $NF
表示列印最後一個欄位,各個欄位預設以空格劃分,可用選項 -F
指定使用什麼字元串來劃分。
LogFile=${UDiskMountDir}/Debug.log
聲明變數 LogFile 並以右邊的內容賦值。將變數 UDiskMountDir 所代表的路徑名與後邊的字元串結合為新的文件名。一般命令中用到的文件名,要不是帶相對路徑的文件名,如 ./diretion/filename
,就是帶絕對路徑的文件名,如 /root/diretion/filename
。
${UDiskMountDir}
表示引用變數 UDiskMountDir 的值。
AllCheckPassed=true
聲明變數 AllCheckPassed 並賦值為true。
AddError()
{
AllCheckPassed=false
echo "-200" > /tmp/VerCheckRes
}
聲明函數 AddError,輸入參數無需聲明。
變數 AllCheckPassed 賦值為true。
echo
命令用於輸出字元串。echo "-200" > /tmp/VerCheckRes
將 -200 輸出到文件 /tmp/VerCheckRes 並覆蓋原有內容。如果將符號 >
換成 >>
則是將內容追加到最後位置。
AddRecord()
{
if [ ! -d "/userdata/recordsDir/" ];then
mkdir -p /userdata/recordsDir/
fi
echo "$*" > /userdata/recordsDir/test
sync
AddDebugLog2UDisk $*
}
聲明函數 AddRecord。
if
表示 if 語句的開始。if[]
方括弧內填判斷條件,如為 true 則執行 then 後的語句,否則跳出 if 語句。fi
表示 if 語句的結束。和 C 語言類似,if 語句也可以有 else
甚至 else if
(shell 內應該寫成 elif
)。if/elif
和 then
如果不在同一行則可以省略符號 ;
。if [ ! -d "/userdata/recordsDir/" ]
判斷目錄 /userdata/recordsDir/ 是否存在,並對結果值取反(!
)。
mkdir
命令用於構建目錄。帶選項 -p
表示構建路徑下所有的目錄。mkdir -p /userdata/recordsDir/
創建目錄 /userdata/recordsDir/, /userdata/ 如果不存在也會被構建。
echo "$*" > /userdata/recordsDir/test
將函數輸入的所有參數都輸出到文件 '/userdata/recordsDir/test' 並覆蓋原有內容。'$*' 表示當前函數或者腳本的所有輸入參數,由於在函數內引用,所以這裡表示函數的所有輸入參數。
sync
命令用於數據同步。Linux 系統中寫入硬碟的數據往往會被先存放於 buffer 中,這樣是為了效率起見,但是如果系統突然斷電,那麼數據就會丟失,這時可以調用 sync 將 buffer 中的數據同步到硬碟。
AddDebugLog2UDisk $*
調用函數 AddDebugLog2UDisk 並傳入當前函數(AddRecord)的所有參數。
AddDebugLog2UDisk()
{
echo "$*" >>"$LogFile"
sync
}
聲明函數 AddDebugLog2UDisk。
echo "\$*" >>"$LogFile"
表示將函數的所有輸入參數輸出到變數 LogFile 表示的文件中,以追加的形式輸入到文件末尾。
sync
同步數據到硬碟。
#生成下/tmp/App/version.txt版本信息
註釋內容
rm -f "$LogFile"
rm
命令用於移除文件。選項 -f
表示強制。rm -f "$LogFile"
強制移除變數 LogFile 代表的文件,變數 LogFile 存儲的是文件名,包括路徑。
echo "0" > /tmp/VerCheckRes
輸出字元串 '0' 到文件 /tmp/VerCheckRes
sh /data/bin/run_normal.sh
在當前的環境中使用另一個shell來執行腳本文件 /data/bin/run_normal.sh,當前環境中變數的值可以在新腳本(/data/bin/run_normal.sh)中被使用(非引用,如果使用 source則變為引用),新腳本文件屬性可以無執行許可權。sh
後邊可以帶選項 -n
、-x
等,-n
用於進行shell腳本的語法檢查,-x
用於實現shell腳本的逐句跟蹤調試並列印該行命令和狀態等。
sleep 2
阻塞當前進程,睡眠 2 秒,和 sleep 2s
同樣效果。sleep 2h
表示睡眠 2 小時。
sh /tmp/App/kill.sh
執行腳本文件 /tmp/App/kill.sh
killall MachineCheckNode
殺掉進程 MachineCheckNode,MachineCheckNode 為進程名。
lastSNScanResult=$(cat /userdata/recordsDir/scan)
讀取文件 /userdata/recordsDir/scan 內容並賦值給變數 lastSNScanResult。
cat
命令用於連接文件並輸出內容到標準輸出。
AddDebugLog2UDisk "last sn scan result:$lastSNScanResult"
調用函數 AddDebugLog2UDisk,並傳遞字元串 last sn scan result:$lastSNScanResult
,$lastSNScanResult
獲取變數 lastSNScanResult 的值然後和前面的字元串結合成新的字元串。
#檢查SN和markData
if [ ! -f /data/bin/sysData/sn ]; then
AddRecord "sn missing"
AddError
else
snContent=$(cat /data/bin/sysData/sn)
AddRecord "sn is:$snContent"
snLen=$(cat /data/bin/sysData/sn | awk '{print length($0)}')
if [ "$snLen" != "16" ] ; then
AddRecord "sn length error $snLen"
AddError
fi
fi
if [ ! -f /data/bin/sysData/sn ]
判斷文件 /data/bin/sysData/sn 是否是常規文件而且存在,'!' 表示對結果值取反。
如果文件 /data/bin/sysData/sn 找不到或者不是常規文件,則調用後邊的語句塊,直到 else 為止。調用函數 AddRecord(傳入字元串 "sn missing") 和 AddError(無參數傳入)。
否則,獲取文件內容並賦值給變數 snContent,調用函數 AddRecord(輸入 sn is:$snContent", 字元串其中會插入變數 snContent 的值);聲明變數 snLen,賦值為命令 cat /data/bin/sysData/sn | awk '{print length($0)}'
的執行結果;判斷 snLen 是否不等於 16,是則調用函數 AddRecord 和 AddError。
$(cat /data/bin/sysData/sn | awk '{print length($0)}')
讀取文件 /data/bin/sysData/sn 的內容,通過管道('|')輸入到後一命令語句,awk 計算輸入內容的第一欄位的字元長度並輸出。
if [ ! -f /data/bin/sysData/markData ]; then
AddRecord "markData missing"
AddError
fi
判斷文件 /data/bin/sysData/markData 是否不存在,不存在則調用後邊的語句(以 fi 為止)。
#檢驗版本信息
while read -r line || [ -n "${line}" ]; do
[ "$line" = "" ] && continue
item_filename=$(echo "$line" | awk -F":" '{print $1}')
item_filemd5=$(echo "$line" | awk -F":" '{print $2}')
if [ ! -e "$item_filename" ]; then
AddRecord "$item_filename missing"
AddError
fi
acture_md5=$(md5sum "$item_filename" | awk '{print $1}')
if [ "$item_filemd5" = "$acture_md5" ]; then
continue
else
AddRecord "$item_filename" "md5 diff! file:" "$item_filemd5" "acture:" "$acture_md5"
AddError
fi
done </tmp/MachineDecDir/check_md5list
while read xxx; do
...
done </filename
迴圈讀取文件名 filename 所指向的文件的內容,每讀取一行存入 xxx 變數。read xxx
為 while 語句的判斷語句,判斷語句和 do
如果不在同一行,則 ;
可省略,如:
while read xxx
do
...
done </filename
read 後邊的 -r
表示讀取內容過程中對特殊字元有效,如 /
(輸入未結束需要換行繼續輸入的特殊符號),'/n' 等等。
[ -n "${line}" ]
判斷 line 的內容不為空。而前邊的 '||' 表示前一語句 read -r line
返回 false 才執行這個語句。
[ "$line" = "" ] && continue
表示如果 line 的內容為空則執行 continue;continue 命令用於跳過當前迴圈內容回去執行 while 判斷語句。
item_filename=$(echo "$line" | awk -F":" '{print $1}')
聲明變數 item_filename 並以後邊語句執行結果賦值。awk -F":" '{print $1}')
將輸入的內容按照 :
來劃分欄位並輸出第1個欄位。awk -F":" '{print $2}')
則是將輸入的內容按照 ':' 來劃分欄位並輸出第2個欄位。
if [ ! -e "$item_filename" ]
用於判斷變數 item_filename 代表的文件名所指的文件是否不存在。!
表示取反。if 判斷的內容為 true 則執行 then 後邊的語句,直到 fi 為止,否則跳過。
md5sum "$item_filename"
表示計算變數 item_filename 代表的文件名所指的文件的MD5值。
| awk '{print $1}'
將前一語句的輸出通過管道連接到後邊的這個語句的輸入,預設按照空格劃分內容並輸出第1個欄位內容。
if [ "$item_filemd5" = "$acture_md5" ]; then
判斷 item_filemd5 和 acture_md5 所代表的內容作為字元串是否一樣,是則執行 then 後的語句。continue 指示程式執行流程直接回到執行 while 判斷語句。else 後的語句塊當 if 的判斷條件不為 true 時執行。
#找到 U 盤路徑
UDiskMountDir=$(df | grep /mnt/sd | awk 'END{print $NF}')
#校驗mcu程式版本,U盤下有mcuversion.txt才校驗
UDiskMountDirfile="${UDiskMountDir}/SpecialDir/mcuversion.txt"
CurMcuVersionFile="/tmp/mcuversion.txt"
if [ -f "$UDiskMountDirfile" ]; then
TargetMcuVersion=$(cat "$UDiskMountDirfile")
if [ -f "$CurMcuVersionFile" ]; then
CurMcuVersion="$(cat "$CurMcuVersionFile")"
if [ "$TargetMcuVersion" != "$CurMcuVersion" ]; then
AddRecord "mcuversion diff cur:$CurMcuVersion target:$TargetMcuVersion"
AddError
fi
else
AddRecord "$CurMcuVersionFile not found"
AddError
fi
fi
UDiskMountDirfile="${UDiskMountDir}/SpecialDir/mcuversion.txt"
聲明變數 UDiskMountDirfile 並賦值為其後的內容,其後是將變數 UDiskMountDir 所代表的路徑名與後邊的字元串結合為新的文件名。
if 語句可以有多重嵌套,如上。
[ "$TargetMcuVersion" != "$CurMcuVersion" ]
判斷 TargetMcuVersion 和 CurMcuVersion 所代表的內容是否不相等。
#寫入檢查結果
if [ $AllCheckPassed = "true" ]; then
AddRecord "Pass"
else
AddRecord "Fail"
fi
if [ $AllCheckPassed = "true" ]; then
判斷 AllCheckPassed 的值是否等於字元串 "true",是則調用函數 AddRecord 並傳入參數字元串 "Pass",否則調用函數 AddRecord 並傳入參數字元串 "Fail"。
#啟動checkall
/data/bin/Factory/MachineCheckNode MachineImcomingTest
調用絕對路徑下的程式 MachineCheckNode,並傳入字元串參數 MachineImcomingTest。此種調用方式,要求程式文件 MachineCheckNode 具有可執行的許可權屬性x。想要查看指定路徑下所有文件或某個文件的屬性可以使用命令 ls -l
查看,r
表示可讀,w
表示可寫,x
表示可執行。
username@DESKTOP-ABCDEF:/mnt/d/username/work/temp$ ls -l
total 0
drwxrwxrwx 1 username username 4096 May 21 16:10 Udisk_IQC
這篇講解到此為止,下期再見!