C++ 條件變量的使用

緒論

并發編程紛繁復雜,其中用于線程同步的主要工具——條件變量,雖然精悍,但是要想正確靈活的運用卻并不容易。
對于條件變量的理解有三個難點:

  1. 為什么wait函數需要將解鎖和阻塞、喚醒和上鎖這兩對操作編程原子的?
  2. 為什么wait函數需要配合while進行使用?
  3. 通知線程是應該先notifyunlock還是先unlocknotify

希望大家看完下面的介紹能夠得到想要的答案。想要了解更多關于C++并發編程信息可以移步的我倉庫:C++并發編程

條件變量

C++提供了兩種條件變量的實現:std::condition_variablestd::condition_variable_any。前者只能和std::mutex配合使用,后者只需要符合互斥的標準即可。因為std::condition_variable_any更通用,所以可能產生額外的開銷,如果沒什么特殊需要,盡可能使用std::condition_variable

條件變量是非常重要的線程同步的手段(目前我認為是最重要的),因此對其的深入理解至關重要。

  • 條件變量總是和互斥一起配合使用,互斥用于保護共享數據,條件變量用于

    1. 通知(通知線程)
    2. 判斷共享數據是否滿足條件(等待線程)
  • 通知線程往往先通過互斥保護共享數據,對數據進行一定的修改后再發送通知(notify_one()、notify_all())。需要注意的是我們應盡可能在臨界區內發送通知,從而避免可能出現的優先級翻轉和條件變量失效問題。雖然臨界區外通知可以讓等待線程一旦被喚醒就能立即解鎖互斥查看是否滿足情況,但是在Pthread進行wait morphint后基本上兩者沒有性能上的差距。詳細的分析可以參考博客:條件變量用例–解鎖與signal的順序問題。

    • notify_one()理論上只會喚醒一個等待線程,適用于共享變量數量發生變化的情況,例如通知消息隊列中的消息個數增加。
    • notify_all()會喚醒所有等待該條件變量的線程,適用于共享變量狀態發生變化的情況,例如通知所有工作線程開始計算。
  • 等待線程先獲得互斥,然后將鎖和判定條件傳遞給wait函數等待返回。

    • wait函數首先會根據判斷條件判斷是否滿足條件(返回true

      • 如果滿足條件,則直接返回(互斥依舊上鎖)

      • 如果不滿足條件,則阻塞等待,并解鎖互斥(讓其他線程得以修改共享數據的狀態)。直到被notify函數喚醒,再次上鎖,判斷條件是否滿足。這里的阻塞和解鎖、喚醒和上鎖都是原子的,就是為了避免兩個動作分別執行出現的條件競態。

        1. 解鎖和阻塞是原子的:lock → !pred() → unlock → sleep;如果變量的改變以及喚醒事件發生在unlock和sleep中間,那么你不會檢測到,也就是錯過了這次喚醒。假如下次喚醒依賴于此次喚醒的成功(也就是說不會主動喚醒第二次),那么將發生死鎖。
        2. 喚醒和上鎖是原子的:wakeup → lock → !pred :如果條件在wakeup和lock之間從滿足變成了不滿足(不是因為其他等待線程修改,而是因為負責喚醒的線程自己再次修改了條件),那么此次喚醒將失敗。假如后面條件的再次滿足依賴于此次條件滿足成功(也就是說條件不會再主動滿足),那么將發生死鎖。

        需要理解的是上面的死鎖的出現是有限定條件的(例如喚醒之間的依賴、條件滿足的依賴),雖然大多數情況下沒有這么嚴格的條件,但是工具本身需要避免這種危險的情況。

        原子操作保證了重要的喚醒和條件滿足都能夠至少被一個等待線程看到。

      • 可以看到wait函數內部需要解鎖互斥,所以就不能使用不提供unlock函數的lock_guard,而應該使用和互斥有相同接口的unique_lock

    • 其實C++的線程庫是對pthread庫的封裝,因此也可以像pthread庫一樣只傳入互斥,解鎖并等待通知,一旦接收到通知后再上鎖,然后在一個while循環中進行判斷。

      while (!pred()) {cond_.wait(lk);  //調用pthread_cond_wait
      }
      

      對于傳入判定條件的版本,其實內部也是這樣的一個封裝罷了。

  • 之所以說notify_one()理論上只會喚醒一個等待線程是因為存在調用一次notify_one()卻喚醒了多個線程的可能性,甚至有時候沒有調用notify等待線程都被喚醒,稱這種意外喚醒等待線程的情況為偽喚醒。按照C++標準的規定,這種偽喚醒出現的數量和頻率都不確定,因此要求等待線程的判定函數不能有副作用(可重用),并且需要在喚醒后再次判斷條件是否滿足,如果不滿足則需要重新等待。這也是為什么上面的代碼使用while進行條件判斷而不是if的原因。

消息隊列

//
// Created by edward on 22-11-16.
// use condtion_variable to genenrate a thread safe message queue
//#include "utils.h"
#include <mutex>
#include <queue>
#include <condition_variable>
#include <iostream>
#include <thread>
#include <string>template<typename T>
class MessageQueue {
public:void push(T t) {std::lock_guard lk(mtx_);       //互斥保護數據queue_.push(std::move(t));cond_.notify_one();				//臨界區內發送通知,避免優先級反轉和條件變量失效}T pop() {T frnt;std::unique_lock lk(mtx_);cond_.wait(lk, [&](){return !queue_.empty();});frnt = std::move(queue_.front());queue_.pop();return frnt;}
private:mutable std::mutex mtx_;mutable std::condition_variable cond_;std::queue<T> queue_;
};using namespace std;template<typename T>
void data_prepare(MessageQueue<T> &messageQueue) {T t;while (cin >> t) {messageQueue.push(std::move(t));}
}template<typename T>
void data_process(MessageQueue<T> &messageQueue) {T t;int idx = 0;while (true) {t = messageQueue.pop(); //數據的處理在臨界區外edward::print("[", idx++, "]:", t);}
}int main() {MessageQueue<string> messageQueue;edward::print("test begin:");thread preparer(data_prepare<string>, ref(messageQueue));thread processer(data_process<string>, ref(messageQueue));preparer.join();//不用等待processer,如果preparer結束,則直接推出進程return 0;
}

運行結果

在這里插入圖片描述

其中用到了我自己寫的庫函數頭文件utils,如果想要了解更多信息可以移步C++ 工具函數庫

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/383503.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/383503.shtml
英文地址,請注明出處:http://en.pswp.cn/news/383503.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

C++Primer學習筆記:第1章 開始

本博客為閱讀《C Primer》&#xff08;第5版&#xff09;的讀書筆記 ps:剛開始的時候我將所有的筆記都放在一篇博客中&#xff0c;等看到第六章的時候發現實在是太多了&#xff0c;導致我自己都不想看&#xff0c;為了日后回顧&#xff08;不那么有心理壓力&#xff09;&#…

【ubuntu】ubuntu14.04上安裝搜狗輸入法

** 在ubuntu14.04.4 desktop 64amd版本上安裝sogou輸入法 ** 0.換安裝源為中國源&#xff08;可選&#xff0c;下載會快些&#xff09; 1.搭fcitx環境 2.安裝sogou for linux 詳細步驟&#xff1a; 因為sogou中文輸入法基于fcitx(Free Chinese Input Toy for X),需要先搭環境…

【ubuntu】ubuntu下用make編譯程序報錯找不到openssl/conf.h

ubuntu下用make編譯程序報錯找不到openssl/conf.h 安裝libssl-dev:i386&#xff0c;sudo apt-get install libssl-dev:i386 看好版本&#xff0c;如果不加i386默認下載的是32位&#xff0c;用ln命令連接過去也還是用不了的!libssl.dev安裝好后&#xff0c;用find / -name libs…

【ubuntu】ubuntu如何改變系統用戶名

ubuntu如何改變系統用戶名 方法1&#xff1a;修改現有用戶名 方法2&#xff1a;創建新用戶&#xff0c;刪掉舊用戶 方法1&#xff1a; * *—&#xff01;&#xff01;&#xff01;有博客說要先改密碼&#xff0c;再改用戶名&#xff0c;否則會出現無法登陸狀況&#xff01;&…

什么是signal(SIGCHLD, SIG_IGN)函數

什么是signal(SIGCHLD, SIG_IGN)函數 在進行網絡編程時候遇到這個函數的使用&#xff0c;自己學習結果如下&#xff0c;有不對請幫忙指正:) signal(SIGCHLD, SIG_IGN)打開manpage康一康~ sighandler_t signal ( int signum, sighandler_t handler );參數1 int signum: 就是…

ssh連接不上linux虛擬機

ssh連接不上linux虛擬機 1.開啟ssh服務 linux虛擬機下命令行輸入&#xff1a; start service ssh如果顯示沒有ssh&#xff0c;就下面兩個試一試哪一個ok&#xff0c;安裝一下ssh&#xff1a; sudo apt-get install openssh-server sudo apt-get install sshd2.還有人說可能是…

沒寫client,想先測試server端怎么辦?

沒寫client&#xff0c;想先測試server端怎么辦&#xff1f; 辦法&#xff1a; 1.先打開終端./server&#xff0c;運行起來server 2.再開一個終端&#xff0c; 輸入nc 127.0.0.1 8888 回車&#xff08;這里port號要和server里邊設置的一致&#xff0c;127.0.0.1是和本機的測試…

【報錯解決】linux網絡編程報錯storage size of ‘serv_addr’ isn’t known解決辦法

linux網絡編程報錯storage size of ‘serv_addr’ isn’t known解決辦法 報錯如下&#xff1a; server.c:18:21: error: storage size of ‘serv_addr’ isn’t known struct sockaddr_in serv_addr, clit_addr; ^server.c:18:32: error: storage size of ‘clit_addr’ isn’…

【c】寫頭文件要加#ifndef,#define, #endif

頭文件首位 編寫.h時&#xff0c; 最好加上如下&#xff0c;用來防止重復包含頭文件&#xff1a; 例如&#xff1a; 要編寫頭文件test.h 在頭文件開頭寫上兩行&#xff1a;#ifndef _TEST_H#define _TEST_H// 文件名的大寫#endif頭文件結尾寫上一行&#xff1a;#endif這樣做是為…

【c】【報錯解決】incompatible implicit declaration

【報錯解決】incompatible implicit declaration 背景; 1.自己封裝的函數wrap.c包含&#xff1a; #include "wrap.h"2.主函數調用如下&#xff1a; #include <stdio.h> #include <stdlib.h> ... #include <errno.h> #include "wrap.h"…

【ubuntu】vim語法高亮設置無效

如果你的.vimrc配置了語法高亮&#xff0c;但是你的vim沒實現&#xff0c;可能你的vim是vim-tiny的黑白版本&#xff0c;你需要vim-gnome這個帶GUI的彩色版本。 apt-get update apt-get upgrade apt-get install vim-gnome reboot打開vi就能看到彩色啦

__attribute__機制介紹

1. __attribute__ GNU C的一大特色&#xff08;卻不被初學者所知&#xff09;就是__attribute__機制。 __attribute__可以設置函數屬性(Function Attribute)、變量屬性(Variable Attribute)和類型屬性(Type Attribute) __attribute__前后都有兩個下劃線&#xff0c;并且后面會緊…

【git】git基本操作命令

1.建立本地倉庫 git config --global user.name "lora" git config --global user.email "xxxgmail.com"2.建立目錄 mkdir xxx3.初始化 cd xxx //進入目錄 git init //初始化4.將代碼上傳至本地緩存區 git add . //上傳全部 git add 文件名 //…

【git】解決gitlab ip更改問題

有時候因為部署gitlab虛擬機的ip發生變化&#xff0c;gitlab的clone地址沒有同時更新到新的ip&#xff0c; 這導致后續clone報錯&#xff0c;解決辦法如下&#xff1a; 進入部署gitlab的主機&#xff1a; sudo vim /opt/gitlab/embedded/service/gitlab-rails/config/gitlab.…

gcc -l參數和-L參數

-l參數就是用來指定程序要鏈接的庫&#xff0c;-l參數緊接著就是庫名&#xff0c;那么庫名跟真正的庫文件名有什么關系呢&#xff1f;就拿數學庫來說&#xff0c;他的庫名是m&#xff0c;他的庫文件名是libm.so&#xff0c;很容易看出&#xff0c;把庫文件名的頭lib和尾.so去掉…

【jenkins】jenkins CI/CD搭建基本過程

1.安裝 &#xff08;1&#xff09;安裝java &#xff08;2&#xff09;安裝jenkins &#xff08;3&#xff09;修改jenkins用戶名密碼配置 &#xff08;4&#xff09;啟動jenkins 2. 插件安裝換源 &#xff08;1&#xff09;插件高級選項換地址 &#xff08;2&#xff09;修改…

apt-get常用命令

一&#xff0c;什么的是apt-get 高級包裝工具&#xff08;英語&#xff1a;Advanced Packaging Tools,簡稱&#xff1a;APT&#xff09;是Debian及其衍生發行版&#xff08;如&#xff1a;ubuntu&#xff09;的軟件包管理器。APT可以自動下載&#xff0c;配置&#xff0c;安裝二…

【jenkins】jenkins build項目的三種方式

jenkins致力于CI/CD&#xff0c; 更改代碼只需要在gitlab push之后&#xff0c;jenkins重新build便可以在tomcat上實現更新部署。 以下為三種構建方式&#xff1a; 1.freestyle project 0. 安裝插件Deploy to container, 并安裝憑證 github連接創建item設置build和post-build …

apt-get 使用詳解

[舉例] 目前常用的 *更新本機中的數據庫緩存&#xff1a; sudo apt-get update *查找包含部分關鍵字的軟件包&#xff1a; sudo apt-cache search <你要查找的name> *安裝指定的軟件&#xff1a; sudo apt-get install <你要安裝的軟件包> *下載軟件包源代碼&…

Buildroot用戶指南

第一章 關于Buildroot Buildroot是一個包含Makefile和修補程序【patch】的集合&#xff0c;這個集合可以使你很容易的為你的目標構建交叉工具鏈【cross-compilationtoolchain】&#xff0c;根文件系統【root filesystem】以及Linux內核映像【kernelimage】。Buildroot可…