C++多線程互斥鎖和條件變量的詳解_第1頁
C++多線程互斥鎖和條件變量的詳解_第2頁
C++多線程互斥鎖和條件變量的詳解_第3頁
C++多線程互斥鎖和條件變量的詳解_第4頁
C++多線程互斥鎖和條件變量的詳解_第5頁
已閱讀5頁,還剩4頁未讀, 繼續(xù)免費閱讀

下載本文檔

版權說明:本文檔由用戶提供并上傳,收益歸屬內容提供方,若內容存在侵權,請進行舉報或認領

文檔簡介

第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. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

最新文檔

評論

0/150

提交評論