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
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...