嘗試閱讀理解一份linux shell腳本

以下內容為本人的學習筆記,如需要轉載,請聲明原文鏈接 微信公眾號「englyf」//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/elifthen 如果不在同一行則可以省略符號 ;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

這篇講解到此為止,下期再見!

Tags: