Shell 腳本中重復(fù)執(zhí)行如何解決,相信很多沒有經(jīng)驗的人對此束手無策,為此本文總結(jié)了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個問題。
創(chuàng)新互聯(lián)是一家專注于網(wǎng)站建設(shè)、成都網(wǎng)站設(shè)計與策劃設(shè)計,榮縣網(wǎng)站建設(shè)哪家好?創(chuàng)新互聯(lián)做網(wǎng)站,專注于網(wǎng)站建設(shè)10余年,網(wǎng)設(shè)計領(lǐng)域的專業(yè)建站公司;建站業(yè)務(wù)涵蓋:榮縣等地區(qū)。榮縣做網(wǎng)站價格咨詢:028-86922220
簡介
flock 是文件鎖命令,它可以保證Linux系統(tǒng)上進程之間安全的訪問臨界資源,在shell腳本中,可以用來控制邏輯的互斥性
實例1
現(xiàn)有腳本 a.sh, 內(nèi)容如下
#!/bin/bash echo "[`date +'%Y-%m-%d %H:%M:%S'`] begin pid:$$..." sleep 10 echo "[`date +'%Y-%m-%d %H:%M:%S'`] end pid:$$..."
在終端(記為終端1)中執(zhí)行 flock -xn ./f.lock -c ./a.sh 命令,結(jié)果如下
[tt@ecs-centos-7 lock_test]$ flock -xn ./f.lock -c ./a.sh [2020-12-10 10:10:45] begin pid:5359... [2020-12-10 10:10:55] end pid:5359...
在上述命令執(zhí)行期間,打開另一個終端(記為終端2),執(zhí)行同樣的命令,結(jié)果如下
[tt@ecs-centos-7 lock_test]$ flock -xn ./f.lock -c ./a.sh [tt@ecs-centos-7 lock_test]$
上面的命令 flock -xn ./f.lock -c ./a.sh 中
-x 選項是排他鎖,有時候也稱為寫鎖,這是默認選項
-n 選項是非阻塞,如果無法獲取鎖,立即返回失敗,而不是一直等待鎖的釋放
-c 選項后面是待執(zhí)行的命令
終端1 中執(zhí)行 flock -xn ./f.lock -c ./a.sh 命令,對 f.lock 文件加鎖,同時執(zhí)行 ./a.sh 命令,執(zhí)行過程會持續(xù)10秒左右( sleep 10 語句 )
由于終端2 中 flock -xn ./f.lock -c ./a.sh 命令是在 終端1 命令執(zhí)行期間執(zhí)行的,此時終端1 還未釋放 f.lock文件鎖,再加上 -n選項是非阻塞的,所以終端2 不會阻塞等待 f.lock 文件鎖,而是立即返回
終端2 如果執(zhí)行 flock -x ./f.lock -c ./a.sh 命令,會一直阻塞等待,直到 終端1 釋放 f.lock 文件鎖,它才會獲取到 f.lock 文件鎖并開始執(zhí) ./a.sh 命令
實例2
實例1 中每次都需要執(zhí)行 flock -xn 文件鎖 -c ./a.sh 命令,而且每個不能重復(fù)執(zhí)行的腳本都要分配一個文件鎖,還得保證不同的腳本得使用不同名字的文件鎖
有沒有辦法做到只要執(zhí)行 ./a.sh 命令就可以實現(xiàn) 實例1 中的功能呢?
答案:有的
我們把 a.sh 稍微修改下,修改之后的內(nèi)容如下
1 #!/bin/bash 2 3 4 echo "[`date +'%Y-%m-%d %H:%M:%S'`] 1111 pid:$$...MY_LOCK:${MY_LOCK}" 5 6 [ "${MY_LOCK}" != "$0" ] && exec env MY_LOCK="$0" flock -xn "$0" "$0" "$@" 7 8 echo "[`date +'%Y-%m-%d %H:%M:%S'`] begin pid:$$...MY_LOCK:${MY_LOCK}" 9 10 sleep 10 11 12 echo "[`date +'%Y-%m-%d %H:%M:%S'`] end pid:$$..."
終端1 執(zhí)行 ./a.sh 命令,輸出如下
[tt@ecs-centos-7 lock_test]$ ./a.sh [2020-12-10 14:11:35] 1111 pid:5944...MY_LOCK: [2020-12-10 14:11:35] 1111 pid:5946...MY_LOCK:./a.sh [2020-12-10 14:11:35] begin pid:5946...MY_LOCK:./a.sh [2020-12-10 14:11:45] end pid:5946...
在終端1 命令執(zhí)行期間,終端2 執(zhí)行 ./a.sh 命令,輸出如下
[tt@ecs-centos-7 lock_test]$ ./a.sh [2020-12-10 14:11:44] 1111 pid:5976...MY_LOCK: [2020-12-10 14:11:44]
新的 a.sh 腳本相比原來新增了第 4、6 兩行
第 4 行是日志打印
第 6 行說明
$0 是腳本名字,這里的值是 ./a.sh
$@ 是傳入 a.sh 腳本的所有參數(shù)
exec 會在當前進程執(zhí)行它后面緊接著的命令,當前腳本進程原來還未執(zhí)行完的命令不會執(zhí)行了
[ "${MY_LOCK}" != "$0" ] 是判斷 MY_LOCK 環(huán)境變量是否和腳本名字( a.sh ) 相同
如果不同,就執(zhí)行 env MY_LOCK="$0" 命令 和 flock -xn "$0" "$0" "$@" 命令
env MY_LOCK="$0" 設(shè)置環(huán)境變量 MY_LOCK 的值為腳本名字
flock -xn "$0" "$0" "$@" 其實就是 flock -xn ./a.sh ./a.sh,它使用當前腳本名字作為文件鎖
實例2 中,執(zhí)行 ./a.sh 命令之后,當運行到第 6 行時,MY_LOCK 變量是空值,所以 [ "${MY_LOCK}" != "$0" ] 的結(jié)果為 true
exec 命令會忽略掉后面未執(zhí)行的命令,也即在當前shell進程中 第 6 行之后的命令都不會執(zhí)行了
緊接著, exec env MY_LOCK="$0" flock -xn "$0" "$0" "$@" 命令, 把 MY_LOCK 變量的值設(shè)置為當前腳本名字 ./a.sh ,同時執(zhí)行 flock -xn "$0" "$0" "$@"命令,此命令會在一個新的子shell中執(zhí)行 ./a.sh ,所以腳本后續(xù)的輸出中打印的進程ID和開始時不一樣
同時,由于在 flock -xn "$0" "$0" "$@" 之前執(zhí)行過 env MY_LOCK="$0",MY_LOCK 變量的值被設(shè)置為了 ./a.sh, 所以 flock -xn "$0" "$0" "$@" 命令重新執(zhí)行 ./a.sh 命令時, 腳本第 6 行的 [ "${MY_LOCK}" != "$0" ] 的結(jié)果為 false, 第 6 行 exec 后面的命令不會執(zhí)行,腳本接著從第 7 行一直執(zhí)行到最后, 結(jié)果輸出 8 和 12 行的日志也說明腳本執(zhí)行完了
看完上述內(nèi)容,你們掌握 Shell 腳本中重復(fù)執(zhí)行如何解決的方法了嗎?如果還想學到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝各位的閱讀!