<二>線程間互斥-mutex互斥鎖和lock_guard

来源:https://www.cnblogs.com/erichome/archive/2022/12/12/16976013.html
-Advertisement-
Play Games

多線程程式 競態條件:多線程程式執行的結果是一致的,不會隨著CPU對線程不同的調用順序而產生不同的運行結果. 解決?:互斥鎖 mutex 經典的賣票問題,三個線程賣100張票 代碼1 #include <iostream> #include <thread> #include <list> #inc ...


多線程程式
競態條件:多線程程式執行的結果是一致的,不會隨著CPU對線程不同的調用順序而產生不同的運行結果.

解決?:互斥鎖 mutex

經典的賣票問題,三個線程賣100張票

代碼1

#include <iostream>
#include <thread>
#include <list>
#include <mutex>
int ticketCount = 100;

std::mutex mtx;//互斥鎖


void sellTicket(int window) {

	while (ticketCount > 0) {
		
		mtx.lock();
		std::cout << "視窗" << window << "銷售" << ticketCount << std::endl;	
		ticketCount--;
		mtx.unlock();
		
		std::this_thread::sleep_for(std::chrono::milliseconds(50));	
		
	}

}//end



int main() {

	std::list<std::thread> tlist;
	
	for (int i = 0; i < 3; i++) {
		tlist.push_back(std::thread(sellTicket,i));
	}

	for (std::thread & t : tlist) {
		t.join();
	}

	system("pause");
	return 0;
}

上面代碼的問題...

while (ticketCount > 0) {
		
    mtx.lock();
    std::cout << "視窗" << window << "銷售" << ticketCount << std::endl;	
    ticketCount--;
    mtx.unlock();		
    std::this_thread::sleep_for(std::chrono::milliseconds(50));	
		
}
如果ticketCount =1 ,當前有一個線程A while (ticketCount > 0)為true,線程A還沒執行ticketCount--完成時,cpu交給了線程B
線程B while (ticketCount > 0)也為true,進入 迴圈體內,造成了買0號票,改進如下

代碼2

#include <iostream>
#include <thread>
#include <list>
#include <mutex>
int ticketCount = 100;

std::mutex mtx;//互斥鎖


void sellTicket(int window) {

	while (ticketCount > 0) {
		
		mtx.lock();
                if(ticketCount >0){
		  std::cout << "視窗" << window << "銷售" << ticketCount << std::endl;	
		  ticketCount--;
                }
		mtx.unlock();		
		std::this_thread::sleep_for(std::chrono::milliseconds(50));			
	}

}//end



int main() {

	std::list<std::thread> tlist;
	
	for (int i = 0; i < 3; i++) {
		tlist.push_back(std::thread(sellTicket,i));
	}

	for (std::thread & t : tlist) {
		t.join();
	}

	system("pause");
	return 0;
}

代碼2還有些問題!! 如下

mtx.lock();
代碼
代碼
代碼
代碼
.....
mtx.unlock();	

如果在代碼lock()和unlock()之間 非常返回,導致mtx沒有正常unlock(),那麼出現死鎖問題 =》智能指針  lock_gurad unique_lock

看lock_gurad

#include <iostream>
#include <thread>
#include <list>
#include <mutex>
int ticketCount = 100;

std::mutex mtx;//互斥鎖


void sellTicket(int window) {

	while (ticketCount > 0) {
		
		{
			std::lock_guard<std::mutex> lock(mtx);
			std::cout << "視窗" << window << "銷售" << ticketCount << std::endl;
			ticketCount--;
			std::this_thread::sleep_for(std::chrono::milliseconds(50));
		
		}
		
	}

}//end



int main() {

	std::list<std::thread> tlist;
	
	for (int i = 0; i < 3; i++) {
		tlist.push_back(std::thread(sellTicket,i));
	}

	for (std::thread & t : tlist) {
		t.join();
	}

	system("pause");
	return 0;
}

上面的圖片中我們知道lock_gurad 的拷貝構造函數被關閉了,所以當我們遇到函數調用需要拷貝構造lock_guard的時候,就有障礙了,這個時候可以用unique_lock

unique_lock 轉移指針,支持帶右值得拷貝賦值,支持參數傳遞拷貝構造的,他的左值的拷貝構造也是被關閉了 看下圖


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

-Advertisement-
Play Games
更多相關文章
  • 關於 SpringBoot 的自動裝配功能,相信是每一個 Java 程式員天天都會用到的一個功能,但是它究竟是如何實現的呢?今天阿粉來帶大家看一下。 自動裝配案例 首先我們通過一個案例來看一下自動裝配的效果,創建一個 SpringBoot 的項目,在 pom 文件中加入下麵的依賴。 <depende ...
  • 這本書是寫什麼的? 這是一本 Go 語言快速入門手冊,目標讀者是有任一編程語言基礎,希望以最快的時間 (比如一個周末) 入門 Go 語言。 這本書應該怎麼讀? 書中幾乎沒有較長篇幅的理論知識,更多的是 示例驅動開發,鼓勵讀者通過快速實踐加深基礎理解,加快學習速度。 全書按照章節順序閱讀即可,其中一些 ...
  • 背景 管理資料庫的界面工具。 開發團隊中一般有非技術背景人員,比如: 產品,功能測試人員; 對他們來說,可能安裝資料庫管理工具客戶端都很麻煩,需要一款線上的網頁工具能方便他們查閱數據。 本地docker安裝 單庫: docker run --name myadmin -d -e PMA_HOST=l ...
  • dubbo當前版本 2.7.3 期望升級到 3.0.11。 升級過程 maven依賴變更 <dependency> <groupId>org.apache.dubbo</groupId> <artifactId>dubbo</artifactId> <version>3.0.11</version> ...
  • 前言 本文給大家分享的是如何通過利用Python實現多功能音樂播放器,廢話不多直接開整~ 開發工具 Python版本: 3.6 相關模塊: os模塊 sys模塊 time模塊 random模塊 PyQt5模塊 環境搭建 安裝Python並添加到環境變數,pip安裝需要的相關模塊即可。 文中完整源碼, ...
  • 官方資料 官方解釋: https://pkg.go.dev/cmd/go#hdr-Build_constraints ,go help buildconstraint 也能看到描述 根據官方描述,go1.16開始建議使用go:build方式,與+build相比更容易被人閱讀。 有關go:build註 ...
  • keepalived 主備使用 本篇主要介紹一下 keepalived 的基本的 主備使用 1.概述 什麼是 keepalived呢,它是一個集群管理中 保證集群高可用的軟體,防止單點故障,keepalived是以VRRP協議為實現基礎的,VRRP全稱Virtual Router Redundanc ...
  • 有了非對稱密鑰、摘要、對稱密鑰等現代密碼學演算法與技術,是不是就能夠保證通信的安全無虞呢,並不是。 密碼學在互聯網應用的四個目標:機密性、完整性、身份驗證、防抵賴。到目前為止,我們討論的技術中,其中防抵賴的目標並沒有達到。 假設A、B、C三個人共用一個對稱加密演算法密鑰,現在A和B互相通信,A和B一直認 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...