版權說明:本文檔由用戶提供并上傳,收益歸屬內容提供方,若內容存在侵權,請進行舉報或認領
文檔簡介
第C++多線程互斥鎖和條件變量的詳解目錄互斥鎖:std::mutex::try_lock條件變量:condition_variable總結我們了解互斥量和條件變量之前,我們先來看一下為什么要有互斥量和條件變量這兩個東西,了解為什么有這兩東西之后,理解起來后面的東西就簡單很多了?。?!
先來看下面這段簡單的代碼:
intg_num=0;
voidprint(intid)
for(inti=0;ii++)
++g_num;
cout"id="id"=="g_numendl;
std::this_thread::sleep_for(std::chrono::seconds(1));
intmain()
threadtha(print,0);
threadthb(print,1);
tha.join();
thb.join();
return0;
}
上述代碼功能大致就是在線程tha和thb中運行函數(shù)print,每個線程對g_num進行加加一次,最后加出來的g_num的值應該是10,那么我們現(xiàn)在來看結果:
我們看到運行結果,為什么打印結果最后,按理來說兩個線程各加五次,最后結果應該是10呀,怎么會是9呢?
如上圖所示,是因為++這個運算符不是原子操作(不會被線程調度機制打斷的操作),我們可以將g_num設置為原子數(shù),改為atomic_intg_num=0;
atomic_intg_num=0;//將g_num設置為原子操作數(shù)
//atomicintg_num=0;這個和上面是一樣的下面這行是模板化之后的
voidprint(intid)
for(inti=0;ii++)
++g_num;
cout"id="id"=="g_numendl;
std::this_thread::sleep_for(std::chrono::seconds(1));
intmain()
threadtha(print,0);
threadthb(print,1);
tha.join();
thb.join();
return0;
}
將g_num設置為原子操作數(shù)之后,在++階段就不會被線程調度機制給打斷,我們來看運行結果:
運行結果是我所期望的但是中間那塊又出了一點小狀況連著打著兩個4,兩個6,這種情況該怎么辦呢?
下面就該說道我們的互斥鎖了:
互斥鎖:
在上述代碼中我們使用了共享資源-----全局量g_num,兩個線程同時對g_num進行++操作,為了保護共享資源,在線程里也有這么一把鎖互斥鎖(mutex),互斥鎖是一種簡單的加鎖的方法來控制對共享資源的訪問,互斥鎖只有兩種狀態(tài),即上鎖(lock)和解鎖(unlock)。
來看下面代碼:
intg_num=0;
std::mutexmtx;//創(chuàng)建鎖對象
voidprint(intid)
for(inti=0;ii++)
mtx.lock();//上鎖
++g_num;
cout"id="id"=="g_numendl;
mtx.unlock();//解鎖
std::this_thread::sleep_for(std::chrono::microseconds(500));
intmain()
threadtha(print,0);
threadthb(print,1);
tha.join();
thb.join();
return0;
}
我們來看運行結果:符合我們最初的預期。
打開官方文檔,可以看到
創(chuàng)建鎖對象只有這個方法,拷貝構造被刪除了。
std::mutex::try_lock
對于互斥鎖的lock和unlock我們都很熟悉了,下面來說一下std::mutex::try_lock這個成員函數(shù)!
try_lock字面意思就是說嘗試上鎖,如果上鎖成功,返回true,上鎖失敗則返回false,但是如果上鎖失敗,他還是會接著往下運行,不會像lock哪運被阻塞在上鎖那塊,所以try_lock必須得在循環(huán)中使用:
intg_num=0;
std::mutexmtx;
voidprint(intid)
for(inti=0;ii++)
mtx.try_lock();//代碼只是將lock換成了try_lock且沒把try_lock扔在循環(huán)中執(zhí)行
++g_num;
cout"id="id"=="g_numendl;
mtx.unlock();
std::this_thread::sleep_for(std::chrono::microseconds(500));
intmain()
threadtha(print,0);
threadthb(print,1);
tha.join();
thb.join();
return0;
}
我們來看運行結果:
unlockofunownedmutex,這玩意思就是說你在給個沒上鎖的互斥鎖解鎖,所以報這錯誤,因此try_lock擱在普通語句中,會有很大的問題,現(xiàn)命我們演示一下將這玩意放到循環(huán)中去弄一邊:
intg_num=0;
std::mutexmtx;
voidprint(intid)
for(inti=0;ii++)
while(!mtx.try_lock())//try_lock失敗時為false前面加了!,所以失敗時為true然后打印嘗試加鎖
{//然后再次嘗試加鎖,只有加鎖成功了,才能出這個while循環(huán)
cout"trylock"endl;
++g_num;
cout"id="id"=="g_numendl;
mtx.unlock();
std::this_thread::sleep_for(std::chrono::microseconds(500));
intmain()
threadtha(print,0);
threadthb(print,1);
tha.join();
thb.join();
return0;
}
我們來看運行結果:
運行結果符合我們的預期,但是try_lock這個函數(shù)有個不好處是太損耗資源了,當它加鎖失敗時,一直嘗試加鎖一直嘗試加鎖,損耗CPU資源。
條件變量:condition_variable
框住這三個函數(shù)較為重要,下面著重來說下面這三個函數(shù):這里順便說一下,下面代碼將會是條件變量和互斥鎖的結合使用,至于為什么要將互斥鎖和條件變量一起使用,原因就是互斥鎖狀態(tài)太單一了,而條件變量允許阻塞,接收信號量等剛好彌補了互斥鎖的缺陷所以這些一起使用!??!
這三個函數(shù)呢,通過一個小實驗來實現(xiàn),通過多線程分別打印123一直到100:
std::mutexmtx;
std::condition_variablecv;
intisReady=0;
constintn=100;
voidprint_A()
std::unique_lockmutexlock(mtx);//unique_lock相當于線程中的智能制造自動解鎖,不需要再unlock
inti=0;
while(in)
while(isReady!=0)
cv.wait(lock);//互斥鎖和信號量一起使用wait參數(shù)為鎖對象
cout"A";
isReady=1;
++i;
std::this_thread::sleep_for(std::chrono::microseconds(100));
cv.notify_all();//當isReady等于0時print_B和print_C處于阻塞狀態(tài)
//該函數(shù)就是喚醒所有等待的函數(shù),然后通過isReady來進行判斷要進行那個函數(shù)的運行
voidprint_B()
std::unique_lockmutexlock(mtx);//unique_lock相當于線程中的智能制造自動解鎖,不需要再unlock
inti=0;
while(in)
while(isReady!=1)
cv.wait(lock);
cout"B";
isReady=2;
++i;
std::this_thread::sleep_for(std::chrono::microseconds(100));
cv.notify_all();
voidprint_C()
std::unique_lockmutexlock(mtx);//unique_lock相當于線程中的智能制造自動解鎖,不需要再unlock
inti=0;
while(in)
while(isReady!=2)
cv.wait(lock);
cout"C";
isReady=0;
++i;
std::this_thread::sleep_for(std::chrono::microseconds(100));
cv.notify_all();
intmain()
threadtha(print_A);
threadthb(print_B);
threadthc(print_C);
tha.join();
thb.join();
thc.join();
return0;
}
上面代碼解析:
運行結果:
我們可以看到上述代碼最后喚醒其他線程使用的是notify_all()函數(shù),notify_all()函數(shù)作用就是環(huán)球其他阻塞的函數(shù),然后因為
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
- 4. 未經(jīng)權益所有人同意不得將文件中的內容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內容本身不做任何修改或編輯,并不能對任何下載內容負責。
- 6. 下載文件中如有侵權或不適當內容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 2024-2025學年山東省菏澤市高二下學期期中考試歷史試題(A)(解析版)
- 2024-2025學年江蘇省鹽城市高二下學期期終考試歷史試題(解析版)
- 2026年生物與醫(yī)學前沿科技知識競賽題集
- 2026年計算機應用基礎初級水平測試題
- 2026年心理學入門認知心理學與社會心理學試題庫
- 2026年城市規(guī)劃領域專業(yè)技術人員考試練習題集
- 2026年文化常識與歷史知識綜合測試題
- 2026年高考化學模擬試題及答案解析
- 2026年寫作技巧基礎訓練初級自測模擬題
- 2026年房地產(chǎn)銷售經(jīng)理人才選拔模擬測試
- 2025-2026學年北京市西城區(qū)高三(上期)期末考試地理試卷(含答案詳解)
- 贛州市章貢區(qū)2026年社區(qū)工作者(專職網(wǎng)格員)招聘【102人】考試參考題庫及答案解析
- 江蘇高職單招培訓課件
- 2026年山東理工職業(yè)學院單招綜合素質考試參考題庫帶答案解析
- 2026年及未來5年市場數(shù)據(jù)中國氟樹脂行業(yè)發(fā)展?jié)摿Ψ治黾巴顿Y方向研究報告
- DB1331∕T 109-2025 雄安新區(qū)建設工程抗震設防標準
- DB37∕T 1317-2025 超細干粉滅火系統(tǒng)技術規(guī)范
- Scratch講座課件教學課件
- 《低碳醫(yī)院評價指南》(T-SHWSHQ 14-2025)
- 2025至2030中國砷化鎵太陽能電池外延片行業(yè)市場深度研究與戰(zhàn)略咨詢分析報告
- 質量環(huán)境及職業(yè)健康安全三體系風險和機遇識別評價分析及控制措施表(包含氣候變化)
評論
0/150
提交評論