c/c++ 多線程 mutex的理解

来源:https://www.cnblogs.com/xiaoshiwang/archive/2018/10/30/9880228.html
-Advertisement-
Play Games

多線程 mutex的理解 mutex,我的理解是每個mutex對象都是一個帶鎖頭的門,這個門有兩個狀態,門開著和門關著,感覺像是廢話。。。 當想查看門的里東西,或者把東西放進門裡,或者從門裡拿出東西前,都需要看看,門是否是打開的。 如果門是打開的,就要進去後趕緊把門關上。關上後,就可以查看屋子裡的東 ...


多線程 mutex的理解

mutex,我的理解是每個mutex對象都是一個帶鎖頭的門,這個門有兩個狀態,門開著和門關著,感覺像是廢話。。。

當想查看門的里東西,或者把東西放進門裡,或者從門裡拿出東西前,都需要看看,門是否是打開的。

  • 如果門是打開的,就要進去後趕緊把門關上。關上後,就可以查看屋子裡的東西,放東西到屋子裡,從屋子裡拿東西。
  • 如果門是關著的,就要在外面等著,直到有人出來時,把門打開了,你才能進去。

每個mutex都是不同的門,當你用mutex a鎖上了一個門,就只能用mutex a去打開,用mutex b是打不開,切記。

例子:用mutex a鎖門,用metex b去開門,結果沒打開,就導致了程式的死鎖。

註意:這個程式專門為了測試,mutex的問題。

#include <list>
#include <iostream>
#include <mutex>
#include <algorithm>
#include <thread>
#include <unistd.h>

using namespace std;

class data_protect{
public:
  list<int> alist{1,2};
  mutex m;
  mutex m1;
public:
  
  void add_list(int val){
    m.lock();  //----------------①
    alist.push_back(val);
  }
  bool contains(int val){
    m1.unlock();//----------------------②
    return find(alist.begin(), alist.end(), val) != alist.end();
  }
};

void func(data_protect& dp){
  dp.add_list(12);
}

int main(){
  data_protect dp;
  thread t(func, ref(dp));
  //t.join();
  t.detach();//---------------③
  //sleep(1);
  dp.add_list(12);//----------------④
  if(dp.contains(12)){//------------------⑤
    cout << "contains 12" << endl;
  }
  for(auto& s : dp.alist){
    cout << s << endl;
  }
  pthread_exit(NULL);

}

執行結果:死鎖,程式永遠在等待鎖的打開。

執行結果分析:

從③處開始就開了一個新的線程a,線程a調用了add_list()方法,add_list方法里,在①處是用m去上的鎖。main函數線程在④處也調用了,add_list()方法,進去後,發現是上鎖的狀態,所以就阻塞在哪裡,等著鎖打開後,main函數線程好進去,然後在⑤處調用了contains方法,contains方法試圖在②處用m1去解m的鎖,所以就解不開①處的鎖,所以就導致了一個線程一直等在①處的鎖的地方,就導致了死鎖。

如果把②處的m1.unlock();換成m.unlock();就能解開鎖了,就不會導致死鎖。

想說明的東西,用哪個mutex上的鎖,就得用哪個mutex去解鎖。

mutex的正確使用方法:不是直接用調用mutex的lock,unlock方法。理由是在lock和unlock中間的某段代碼如果崩潰掉,就會導致unlock方法沒有被執行,也就導致了,鎖沒有解開,別線程再來訪問時,就變成了死鎖。

所以使用:std::lock_guard<std::mutex>,它的好處是,即使發生了異常也能自動解鎖。

例子:

#include <list>
#include <iostream>
#include <mutex>
#include <algorithm>
#include <thread>
#include <unistd.h>

using namespace std;

class data_protect{
public:
  list<int> alist{1,2};
  mutex m;
public:
  
  void add_list(int val){
    lock_guard<mutex> g(m);
    alist.push_back(val);
  }
  bool contains(int val){
    lock_guard<mutex> g(m);
    return find(alist.begin(), alist.end(), val) != alist.end();
  }
};

void func(data_protect& dp){
  dp.add_list(12);
}

int main(){
  data_protect dp;
  thread t(func, ref(dp));
  //t.join();
  t.detach();
  //sleep(1);
  dp.add_list(12);
  if(dp.contains(12)){
    cout << "contains 12" << endl;
  }
  for(auto& s : dp.alist){
    cout << s << endl;
  }
  pthread_exit(NULL);

}

c/c++ 學習互助QQ群:877684253

本人微信:xiaoshitou5854


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 多線程 繞過mutex的保護 mutex,能夠解決線程安全的問題,但它不是萬能的。下麵的例子雖然使用了mutex,但是惡意註入了一個外部函數,導致把被mutex保護的雙向鏈表,讓一個外部的指針指向了,結果就可以通過這個外部的指針操作被保護的雙向鏈表,也就繞過了metex的保護。 例子: c++ in ...
  • JDBC 筆記 作者:晨鐘暮鼓c個人微信公眾號:程式猿的月光寶盒 對應pdf版:https://download.csdn.net/download/qq_22430159/10754554 沒有積分的可關註公眾號後臺回覆[JDBC] Day1 JDBC概述+JDBC完成CRUD+DAO設計 1.J ...
  • 1. 函數進階 動態接收位置參數 之前寫的函數都是固定參數的,假設有個函數需要的參數由幾十個,一個個寫在形參的位置會非常麻煩,因此我們要考慮使用動態參數,使用動態參數時需要在參數前加 ,表示接收多個參數: 從上面的例子我們可以看出,動態參數可以接收任意個參數,在形參中作為一個元組的形式傳遞過來;但是 ...
  • 利用tensorflow實現數據的線性回歸 導入相關庫 import tensorflow as tf import numpy import matplotlib.pyplot as plt rng = numpy.random 參數設置 learning_rate = 0.01 training ...
  • tensorflow常數操作 結果 a=2, b=3 Addition with constants: 5 Multiplication with constants: 6 tensorflow變數操作 變數作為圖形輸入,構造器的返回值作為變數的輸出,在運行會話時,傳入變數的值,在進行運算。 結果 ...
  • 線程的狀態 線程的所有狀態在Thread中的State枚舉中定義 public enum State{ NEW, //剛剛新建的線程,還沒有開始執行 RUNNABLE, //執行時的狀態 BLOCKED, //在執行過程中遇到synchronized同步塊,進入blocked阻塞狀態,暫停執行,直到 ...
  • 前通過傳智的視頻自學了webservice的基本使用,也瞭解到webservice就是一種跨編程語言和跨操作系統平臺的遠程調用技術。 對於這些理論知識在這裡也不再做過多的解釋,本次主要就是記錄與分享使用cxf 框架完成遠程調用氣象局提供的介面,來實現天氣查詢的全過程。 1、項目搭建 * 創建一個ma ...
  • 服務發佈者 在服務發佈者的springboot主配置文件application.properties中添加dubbo配置 服務調用者 在服務調用者的springboot主配置文件application.properties中添加dubbo配置 應用配置參數(必須配置) 服務掃描的包 註冊中心支持的配 ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...