每日一句英語學(xué)習(xí),每天進步一點點:
"Better not to ignore the past but learn from it instead. Otherwise, history has a way of repeating itself."
「最好不要無視過去,而是從中汲取經(jīng)驗教訓(xùn),否則,歷史會有重演的時候?!?/span>
前言
有某些場景下,我們不希望有多個相同的 Linux 進程 或 Shell 腳本同時執(zhí)行,因為相同進程同時執(zhí)行,可能會破壞數(shù)據(jù)的一致性。
當然還有在 C++ 代碼里,有時希望保證程序中一個類只有一個實例,并提供一個訪問它的全局訪問點,也就是所謂的「單例模式」。只有一個實例很重要,比如一個打印機可以有多個打印任務(wù),但是只有一個正在工作的任務(wù),一個系統(tǒng)只能有一個窗口管理器或文件系統(tǒng)。
接下來,簡單介紹下:
Linux 命令的方式控制進程是「單例」的方式;
C 代碼單進程控制的實現(xiàn);
C++ 線程安全的「單例模式」實現(xiàn)。
正文
flock 命令為腳本加鎖
可以用flock
命令為 Shell 腳本加鎖。當多個進程可能會執(zhí)行同一個腳本,這些進程需要保證其它進程沒有在操作,以免重復(fù)執(zhí)行。通常,這樣的進程會使用一個「鎖文件」,也就是建立一個文件來告訴別的進程自己在運行,如果檢測到那個文件存在則認為有操作同樣數(shù)據(jù)的進程在工作。
flock命令來為腳本加鎖,如下命令:
flock -xn <鎖文件> -c <shell腳本>
-x : 獲取一個排它鎖,或者稱為寫入鎖,為默認項
-n : 非阻塞模式,當獲取鎖失敗時,返回 1 而不是等待
-c : 執(zhí)行命令或腳本
實戰(zhàn)演示
1. 編寫一個測試腳本 test.sh
#! /bin/bash
echo "Hello World"
sleep 1000
2. flock
命令給腳本加鎖
flock -xn ./test.lock -c "/root/test.sh"
3. 開啟另外一個 bash 窗口運行同個的腳本
另外一個 bash 窗口運行了同個腳本后,未獲取到鎖直接返回了,直到上一個腳本運行完畢,這個才可以開始正常運行。
應(yīng)用的場景
可以在 Linux 定時器/etc/crontab
里運用flock
命令為腳本加鎖,防止重復(fù)執(zhí)行:
* * * * * (flock -xn ./test.lock -c "/root/test.sh")
C 代碼實現(xiàn)單進程控制
通常后臺服務(wù)器程序都必須有且只有一個進程,那么如何控制單進程呢?思想和上面提到的flock
命令差不多。
我們可以通過flock
系統(tǒng)接口函數(shù)對某個文件進行加鎖
若加鎖不正常,說明后臺服務(wù)進程已經(jīng)在運行了,這時則直接報錯退出;
若加鎖成功,說明后臺服務(wù)進程沒有在運行,這時可以正常啟用進程。
用 flock 函數(shù)實現(xiàn)的單進程控制代碼
實戰(zhàn)演練
我們在 main
函數(shù)使用上面的函數(shù):
int main(void)
{
//進程單實例運行檢測
if(0 != server_is_running())
{
printf("myserver process is running!!!!! Current process will exit !\n");
return -1;
}
while(1)
{
printf("myserver doing ... \n");
sleep(2);
}
return 0;
}
運行程序,可知進程pid是 6965
[root@lincoding singleprocess]# ./myserver
server is not running! begin to run..... pid=6965
myserver doing ...
myserver doing ...
此時,再運行同個程序,這時會報錯退出,因為檢測到程序已經(jīng)在運行中,不可以起另外一個進程。
[root@lincoding singleprocess]# ./myserver
server is runing now! errno=11
myserver process is running!!!!! Current process will exit !
C++ 單例模式
單例模式指在整個系統(tǒng)生命周期里,保證一個類只能產(chǎn)生一個實例,確保該類的唯一性。
單例類的特點:
聲明「構(gòu)造函數(shù)和析構(gòu)函數(shù)」為 private 類型,目的禁止外部構(gòu)造和析構(gòu)
聲明「復(fù)制構(gòu)造和賦值操作」函數(shù)為 private 類型,目的是禁止外部拷貝和賦值,確保實例的唯一性
類里有個獲取實例的「靜態(tài)函數(shù)」,可以全局訪問
還有需要注意的是寫單例類時,要注意多線程的競爭的問題,因為可能存在當兩個線程同時獲取單例對象時,產(chǎn)生出了兩個對象,這就違背了單例模式的唯一性。
單例模式實現(xiàn)的方式有很多種,這里推薦一下相對比較簡潔的懶漢式單例的兩種寫法:
在 C++ 11 標準中提出「局部靜態(tài)變量」初始化具有線程安全性,那么此時寫出一個線程安全的單例類,只需要幾行代碼。
Single 使用的靜態(tài)變量是一個「局部靜態(tài)變量」,因此只有在 Single 的GetInstance()
函數(shù)被調(diào)用時其才會被創(chuàng)建,從而擁有了延遲初始化(Lazy)的效果,提高了程序的啟動性能。同時該實例將生存至程序執(zhí)行完畢。而就 Single 的用戶代碼而言,其生存期貫穿于整個程序生命周期,從程序啟動開始直到程序執(zhí)行完。
同時,C++ 11 也提供一個新的東西叫
std::call_once
,配合std::once_flag
,可以保證函數(shù)在任何情況下只調(diào)用一次。
小結(jié)
推薦閱讀:
「C++ 篇」答應(yīng)我,別再if/else走天下了可以嗎
關(guān)注公眾號,后臺回復(fù)「我要學(xué)習(xí)」,即可免費獲取精心整理「服務(wù)器 Linux C/C++ 」成長路程(書籍資料 + 思維導(dǎo)圖)
免責聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺僅提供信息存儲服務(wù)。文章僅代表作者個人觀點,不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!