成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

如何解決 Shell 腳本重復執行的問題

系統 Linux
在開發過程中,經常會使用shell腳本去完成定時備份的任務,普遍的做法是通過系統的定時任務定時執行備份腳本.

 [[357358]]

本文轉載自微信公眾號「Linux開發那些事兒」,作者LinuxThings。轉載本文請聯系Linux開發那些事兒公眾號。   

在開發過程中,經常會使用shell腳本去完成定時備份的任務,普遍的做法是通過系統的定時任務定時執行備份腳本

設想這樣一種場景,本次備份時間到了,自動執行備份腳本,如果備份比較耗時的話,會一直持續到下一次備份時間到了還未結束,下次備份又會自動調用備份腳本,相當于同一時間有兩個進程在執行備份腳本,這可能會導致備份數據錯亂或其他不可預知的問題

更進一步,假如執行備份腳本消耗的時間遠大于設置的備份間隔的話,系統會出現多個同時在執行腳本的Bash實例,會占用大量的系統資源,進而影響正常業務程序的運行

那如何解決上述shell腳本重復執行的問題呢,本文將要介紹的 flock 命令可以解決這個問題

簡介

flock 是文件鎖命令,它可以保證Linux系統上進程之間安全的訪問臨界資源,在shell腳本中,可以用來控制邏輯的互斥性

實例1

現有腳本 a.sh, 內容如下

  1. #!/bin/bash 
  2.  
  3. echo "[`date +'%Y-%m-%d %H:%M:%S'`] begin pid:$$..." 
  4.  
  5. sleep 10 
  6.  
  7. echo "[`date +'%Y-%m-%d %H:%M:%S'`] end pid:$$..." 

在終端(記為終端1)中執行 flock -xn ./f.lock -c ./a.sh 命令,結果如下

  1. [tt@ecs-centos-7 lock_test]$ flock -xn ./f.lock -c ./a.sh  
  2. [2020-12-10 10:10:45] begin pid:5359... 
  3. [2020-12-10 10:10:55] end pid:5359... 

在上述命令執行期間,打開另一個終端(記為終端2),執行同樣的命令,結果如下

  1. [tt@ecs-centos-7 lock_test]$ flock -xn ./f.lock -c ./a.sh  
  2. [tt@ecs-centos-7 lock_test]$ 

上面的命令 flock -xn ./f.lock -c ./a.sh 中

-x 選項是排他鎖,有時候也稱為寫鎖,這是默認選項

-n 選項是非阻塞,如果無法獲取鎖,立即返回失敗,而不是一直等待鎖的釋放

-c 選項后面是待執行的命令

終端1 中執行 flock -xn ./f.lock -c ./a.sh 命令,對 f.lock 文件加鎖,同時執行 ./a.sh 命令,執行過程會持續10秒左右( sleep 10 語句 )

由于終端2 中 flock -xn ./f.lock -c ./a.sh 命令是在 終端1 命令執行期間執行的,此時終端1 還未釋放 f.lock文件鎖,再加上 -n選項是非阻塞的,所以終端2 不會阻塞等待 f.lock 文件鎖,而是立即返回

終端2 如果執行 flock -x ./f.lock -c ./a.sh 命令,會一直阻塞等待,直到 終端1 釋放 f.lock 文件鎖,它才會獲取到 f.lock 文件鎖并開始執 ./a.sh 命令

實例2

實例1 中每次都需要執行 flock -xn 文件鎖 -c ./a.sh 命令,而且每個不能重復執行的腳本都要分配一個文件鎖,還得保證不同的腳本得使用不同名字的文件鎖

有沒有辦法做到只要執行 ./a.sh 命令就可以實現 實例1 中的功能呢?

答案:有的

我們把 a.sh 稍微修改下,修改之后的內容如下

  1. 1 #!/bin/bash 
  2.  2  
  3.  3  
  4.  4 echo "[`date +'%Y-%m-%d %H:%M:%S'`] 1111 pid:$$...MY_LOCK:${MY_LOCK}" 
  5.  5  
  6.  6 [ "${MY_LOCK}" != "$0" ] && exec env MY_LOCK="$0" flock -xn "$0" "$0" "$@" 
  7.  7  
  8.  8 echo "[`date +'%Y-%m-%d %H:%M:%S'`] begin pid:$$...MY_LOCK:${MY_LOCK}" 
  9.  9  
  10. 10 sleep 10 
  11. 11  
  12. 12 echo "[`date +'%Y-%m-%d %H:%M:%S'`] end pid:$$..." 

終端1 執行 ./a.sh 命令,輸出如下

  1. [tt@ecs-centos-7 lock_test]$ ./a.sh 
  2. [2020-12-10 14:11:35] 1111 pid:5944...MY_LOCK: 
  3. [2020-12-10 14:11:35] 1111 pid:5946...MY_LOCK:./a.sh 
  4. [2020-12-10 14:11:35] begin pid:5946...MY_LOCK:./a.sh 
  5. [2020-12-10 14:11:45] end pid:5946... 

在終端1 命令執行期間,終端2 執行 ./a.sh 命令,輸出如下

  1. [tt@ecs-centos-7 lock_test]$ ./a.sh 
  2. [2020-12-10 14:11:44] 1111 pid:5976...MY_LOCK: 
  3. [2020-12-10 14:11:44] 

新的 a.sh 腳本相比原來新增了第 4、6 兩行

第 4 行是日志打印

第 6 行說明

$0 是腳本名字,這里的值是 ./a.sh

$@ 是傳入 a.sh 腳本的所有參數

exec 會在當前進程執行它后面緊接著的命令,當前腳本進程原來還未執行完的命令不會執行了

[ "${MY_LOCK}" != "$0" ] 是判斷 MY_LOCK 環境變量是否和腳本名字( a.sh ) 相同

如果不同,就執行 env MY_LOCK="$0" 命令 和 flock -xn "$0" "$0" "$@" 命令

env MY_LOCK="$0" 設置環境變量 MY_LOCK 的值為腳本名字

flock -xn "$0" "$0" "$@" 其實就是 flock -xn ./a.sh ./a.sh,它使用當前腳本名字作為文件鎖

實例2 中,執行 ./a.sh 命令之后,當運行到第 6 行時,MY_LOCK 變量是空值,所以 [ "${MY_LOCK}" != "$0" ] 的結果為 true

exec 命令會忽略掉后面未執行的命令,也即在當前shell進程中 第 6 行之后的命令都不會執行了

緊接著, exec env MY_LOCK="$0" flock -xn "$0" "$0" "$@" 命令, 把 MY_LOCK 變量的值設置為當前腳本名字 ./a.sh ,同時執行 flock -xn "$0" "$0" "$@"命令,此命令會在一個新的子shell中執行 ./a.sh ,所以腳本后續的輸出中打印的進程ID和開始時不一樣

同時,由于在 flock -xn "$0" "$0" "$@" 之前執行過 env MY_LOCK="$0",MY_LOCK 變量的值被設置為了 ./a.sh, 所以 flock -xn "$0" "$0" "$@" 命令重新執行 ./a.sh 命令時, 腳本第 6 行的 [ "${MY_LOCK}" != "$0" ] 的結果為 false, 第 6 行 exec 后面的命令不會執行,腳本接著從第 7 行一直執行到最后, 結果輸出 8 和 12 行的日志也說明腳本執行完了

總結

實例1 和 實例2 提供了兩種解決 腳本重復執行的 方式,主要都是利用 flock 命令設置文件鎖來實現的,實例2 的方式更簡單,只需要在腳本開頭加上 [ "${MY_LOCK}" != "$0" ] && exec env MY_LOCK="$0" flock -xn "$0" "$0" "$@" 語句,調用腳本的命令保持不變

更多關于 flock 命令的選項及用法可以通過 man flock 自行查看

 

責任編輯:武曉燕 來源: Linux開發那些事兒
相關推薦

2012-09-05 11:09:15

SELinux操作系統

2010-10-27 15:40:14

oracle分頁查詢

2017-10-17 09:21:06

2010-04-29 17:46:31

Oracle死鎖

2019-11-26 14:30:20

Spring循環依賴Java

2024-12-05 09:06:58

2023-07-18 16:05:00

IP地址

2009-09-21 17:10:14

struts Hibe

2021-10-20 20:27:55

MySQL死鎖并發

2011-08-29 10:34:00

網絡安全云安全云計算

2023-10-30 18:35:47

MySQL主從延時

2021-06-06 13:05:15

前端跨域CORS

2011-03-23 14:42:47

CPU過度消耗

2010-07-16 13:52:26

telnet漏洞

2017-07-20 07:30:16

大數據數據互聯網

2024-10-29 16:41:24

SpringBoot跨域Java

2013-05-21 10:49:59

Windows硬件沖突

2017-09-23 22:07:24

深度學習N 體問題GAN

2022-11-04 08:00:00

2011-03-29 13:29:10

點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 自拍 亚洲 欧美 老师 丝袜 | 国产精品亚洲综合 | 亚洲精品综合一区二区 | 国产精品久久久久久 | 亚洲精品国产电影 | 色av一区二区三区 | 97久久精品午夜一区二区 | 天天爱爱网 | 日韩在线中文字幕 | av在线免费观看网站 | 亚洲精品国产成人 | 超碰美女在线 | 久久精品一 | 在线国产一区二区 | 激情av在线 | 欧美国产视频 | 久久亚洲春色中文字幕久久久 | 久久久一二三区 | 亚洲成av片人久久久 | 欧美性tv | 成人av一区二区三区 | 色婷婷综合久久久久中文一区二区 | 欧美乱人伦视频 | 99视频在线播放 | 亚洲一级毛片 | 欧美亚洲国产一区二区三区 | 一级黄色大片 | 三级视频在线观看 | 国产一区二区在线免费观看 | 九九热国产精品视频 | 狠狠爱综合 | www.嫩草| 精品福利一区二区三区 | 久久99视频| 日韩精品一区二区三区在线播放 | 日韩小视频在线 | 久久久91精品国产一区二区三区 | 伊色综合久久之综合久久 | 酒色成人网 | 99国产视频| 国产欧美精品一区 |