c++中的mutex同步機制與多線程同步實現

C++ 中的 std::mutex 與多線程同步

在多線程編程中,互斥鎖(Mutex) 是一種同步機制,用于保護共享資源(如變量、數據結構)免受數據競爭(Data Race)的影響。C++ 標準庫中的 std::mutex 提供了基本的互斥功能,常與 std::lock_guardstd::unique_lock 配合使用,以實現線程安全的資源訪問。


1. std::mutex 的基本概念

  • 作用:確保在同一時刻只有一個線程可以訪問共享資源。
  • 頭文件#include <mutex>
  • 常用方法
    • lock():鎖定互斥鎖,若已被鎖定則阻塞等待。
    • unlock():釋放互斥鎖。
    • try_lock():嘗試鎖定互斥鎖,若失敗則立即返回(不阻塞)。

2. 示例:不使用互斥鎖導致數據競爭

場景:多個線程對共享計數器進行遞增操作。
#include <iostream>
#include <thread>
#include <vector>int counter = 0;void increment() {for (int i = 0; i < 100000; ++i) {++counter;  // 非原子操作,存在數據競爭}
}int main() {std::vector<std::thread> threads;for (int i = 0; i < 10; ++i) {threads.emplace_back(increment);}for (auto& t : threads) {t.join();}std::cout << "Final counter value: " << counter << std::endl;return 0;
}

輸出(每次運行結果可能不同):

Final counter value: 789123  // 錯誤值(期望 1,000,000)

問題分析

  • ++counter 不是原子操作,多個線程同時修改 counter 導致數據競爭。
  • 結果不可預測,可能遠小于預期值。

3. 使用 std::mutexstd::lock_guard 修復數據競爭

改進方案:通過互斥鎖保護共享資源。
#include <iostream>
#include <thread>
#include <vector>
#include <mutex>  // 引入 mutexint counter = 0;
std::mutex mtx;  // 定義互斥鎖void increment() {for (int i = 0; i < 100000; ++i) {mtx.lock();     // 加鎖++counter;      // 安全訪問共享資源mtx.unlock();   // 解鎖}
}int main() {std::vector<std::thread> threads;for (int i = 0; i < 10; ++i) {threads.emplace_back(increment);}for (auto& t : threads) {t.join();}std::cout << "Final counter value: " << counter << std::endl;return 0;
}

輸出

Final counter value: 1000000

關鍵點

  • 互斥鎖保護臨界區mtx.lock()mtx.unlock() 之間的代碼為臨界區,確保一次只有一個線程執行。
  • 避免死鎖:必須成對調用 lock()unlock(),否則可能導致死鎖。

4. 使用 std::lock_guard 自動管理鎖的生命周期

改進方案:使用 RAII(Resource Acquisition Is Initialization) 模式自動加鎖/解鎖。
#include <iostream>
#include <thread>
#include <vector>
#include <mutex>int counter = 0;
std::mutex mtx;void increment() {for (int i = 0; i < 100000; ++i) {std::lock_guard<std::mutex> lock(mtx);  // 構造時加鎖,析構時自動解鎖++counter;  // 安全訪問共享資源}
}int main() {std::vector<std::thread> threads;for (int i = 0; i < 10; ++i) {threads.emplace_back(increment);}for (auto& t : threads) {t.join();}std::cout << "Final counter value: " << counter << std::endl;return 0;
}

輸出

Final counter value: 1000000

優勢

  • 自動解鎖lock_guard 在作用域結束時自動調用 unlock(),避免手動管理鎖。
  • 異常安全:即使發生異常,lock_guard 也會確保解鎖。

5. std::unique_lock:更靈活的鎖管理

使用場景:需要延遲鎖定、條件變量或動態鎖定策略。
#include <iostream>
#include <thread>
#include <mutex>std::mutex mtx;
int shared_data = 0;void access_data() {std::unique_lock<std::mutex> lock(mtx, std::defer_lock); // 不立即加鎖// 做一些不需要鎖的操作...lock.lock();  // 顯式加鎖shared_data = 42;std::cout << "Data updated: " << shared_data << std::endl;// lock 在作用域結束時自動解鎖
}int main() {std::thread t(access_data);t.join();return 0;
}

輸出

Data updated: 42

特點

  • 延遲鎖定:使用 std::defer_lock 參數延遲加鎖。
  • 支持移動:可以將鎖的所有權轉移到其他線程。
  • 配合條件變量:常用于 std::condition_variable 的等待操作。

6. 互斥鎖的最佳實踐

原則說明
保護共享資源所有對共享資源的訪問都必須通過互斥鎖保護。
最小化臨界區鎖定時間越短越好,避免阻塞其他線程。
避免死鎖按固定順序加鎖,不嵌套加鎖。
使用 RAII 風格優先使用 lock_guardunique_lock,避免手動調用 lock()/unlock()
避免遞歸加鎖默認 std::mutex 不支持遞歸加鎖,多次調用 lock() 會導致死鎖。

7. 總結

  • 互斥鎖(std::mutex 是 C++ 多線程編程中保護共享資源的核心工具。
  • std::lock_guard 提供自動加鎖/解鎖,適合大多數同步場景。
  • std::unique_lock 提供更靈活的鎖管理,適用于復雜同步需求(如條件變量)。
  • 關鍵點:確保所有共享資源訪問都通過互斥鎖保護,避免數據競爭和死鎖。

通過合理使用互斥鎖,可以編寫出高效、安全的多線程程序。

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

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

相關文章

網絡安全2023—新安全新發展

關于綠盟科技 綠盟科技集團股份有限公司(以下簡稱綠盟科技),成立于 2000 年 4 月,總部位于北京。公司于 2014 年 1 月 29 日在深圳證券交易所創業板上市,證券代碼:300369。綠盟科技在國內設有 50余個分支機構,為政府、金融、運營商、能源、交通、科教文衛等行業用戶與各…

WebSocket掃盲

WebSocket 是一種網絡通信協議&#xff0c;它允許在單個 TCP 連接上進行全雙工、雙向的實時通信。它是為了解決傳統 HTTP 協議在實時交互應用中的局限性而設計的。 核心概念和特點 解決 HTTP 的痛點&#xff1a; 單向性&#xff1a; HTTP 是請求-響應模式。客戶端發起請求&…

Springboot整合高德地圖

1.登錄高德開放平臺 高德開放平臺 | 高德地圖API 2.獲取密鑰key 1.點擊控制臺 2.創建新應用 3.添加key 4.創建key 5.獲取key 3.java整合 1.高德配置類 package com.thk.controller.map;import org.springframework.beans.factory.annotation.Value; import org.springfram…

【SQL知識】PDO 和 MySQLi 的區別

目錄 簡介 主要區別 預處理語句示例比較 PDO 示例 MySQLi 示例 選擇建議 簡介 PDO (PHP Data Objects) 和 MySQLi (MySQL Improved) 都是 PHP 中用于數據庫操作的擴展&#xff0c;都支持預處理語句&#xff0c;但有一些重要區別&#xff1a; 主要區別 數據庫支持 PDO&am…

python打卡 DAY 45 Tensorboard使用介紹

目錄 一、TensorBoard 發展歷史與原理 1. 演進歷程 2. 核心架構原理 二、TensorBoard 核心功能操作 1. 基礎配置方法 2. 常用功能速查表 三、CIFAR10 實戰演示 1. MLP 模型監控配置 2. CNN 特征可視化 四、TensorBoard 高級功能 1. 超參數調優 2. 3D點云可視化 五、…

Swift 中 Result 類型全解析:從基礎到進階

在現代 iOS 開發中&#xff0c;Swift 的 Result 類型是處理同步與異步錯誤的一大利器。相比傳統的 throws / do-catch 語法&#xff0c;它更清晰、結構化&#xff0c;也更易于組合式編程。 本文將帶你從 Result 的基礎定義出發&#xff0c;逐步深入其在實際項目中的多種應用&am…

Github 2025-06-28 Rust開源項目日報 Top10

根據Github Trendings的統計,今日(2025-06-28統計)共有10個項目上榜。根據開發語言中項目的數量,匯總情況如下: 開發語言項目數量Rust項目10Rust實現的非官方Bitwarden兼容服務器 創建周期:2317 天開發語言:Rust協議類型:GNU Affero General Public License v3.0Star數量…

python 寫一個判斷文本中是否有手機號的函數,并提取出文本中的手機號

我們需要判斷文本中是否有手機號&#xff0c;并提取出手機號。 中國大陸的手機號規則&#xff1a; 1. 通常為11位數字。 2. 目前手機號段分配如下&#xff1a; - 移動號段&#xff1a;134(0-8)、135、136、137、138、139、147、148、150、151、152、157、158、159、172、178、1…

作物生長模型Oryza V3實戰12:drate程序詳解

drate(v2).exe,可以通過觀察移植日、穗部分化、開花和成熟的物候日期(即日和年),DRATE(v2)用于校準四個階段的發展速率:幼苗期(DVRJ,oCday-1)、光周期敏感期(DVRI,oCday-1)、穗部發育期(DVRP,oCday-1)和生殖期(DVRR,oCday-1)。 一 準備輸入文件 1、準備.crp,.…

利用視覺-語言模型搭建機器人靈巧操作的支架

25年6月來自斯坦福和德國卡爾斯魯厄理工的論文“Scaffolding Dexterous Manipulation with Vision-Language Models”。 靈巧機械手對于執行復雜的操作任務至關重要&#xff0c;但由于演示收集和高維控制的挑戰&#xff0c;其訓練仍然困難重重。雖然強化學習 (RL) 可以通過在模…

面試拷打-20250701

memcopy和memmov 詳細解釋 示例1&#xff1a;不重疊的內存區域 正常復制。 示例2&#xff1a;重疊的內存區域 原始數據&#xff1a;src2是一個包含字符串"HelloWorld"的字符數組。使用memcpy&#xff1a; memcpy(src2 2, src2, 5);試圖將src2中的前5個字符復制…

什么是 BigKey?

Redis BigKey 深度解析&#xff1a;識別、危害與優化方案 什么是 BigKey&#xff1f; 在 Redis 中&#xff0c;BigKey 是指存儲大量數據的單個鍵&#xff0c;這些鍵通常具有異常大的內存占用或包含大量元素。BigKey 不是由數據類型定義&#xff0c;而是由其資源消耗決定的。 …

量化選股策略 聚寬

# 量化選股策略完整分析與優化建議 ## 策略整體架構分析 這個量化交易策略主要由以下幾個核心部分組成&#xff1a; 1. **初始化設置**&#xff1a;配置基準指數、交易參數和全局變量 2. **選股邏輯**&#xff1a;通過財務指標篩選優質股票 3. **股票過濾**&#xff1a;排除…

Python 數據分析:numpy,抽提,布爾索引2。

目錄 1 示例代碼2 歡迎糾錯3 論文寫作/Python 學習智能體------以下關于 Markdown 編輯器新的改變功能快捷鍵合理的創建標題&#xff0c;有助于目錄的生成如何改變文本的樣式插入鏈接與圖片如何插入一段漂亮的代碼片生成一個適合你的列表創建一個表格設定內容居中、居左、居右S…

解決leetcode第3597題分割字符串

3597. 分割字符串 難度&#xff1a;中等 問題描述&#xff1a; 給你一個字符串 s&#xff0c;按照以下步驟將其分割為 互不相同的段 &#xff1a; 從下標 0 開始構建一個段。 逐字符擴展當前段&#xff0c;直到該段之前未曾出現過。 只要當前段是唯一的&#xff0c;就將其…

電源芯片之DCDC初探索ING

1. 概述 DC-DC轉換器的意思是直流變直流&#xff08;不同的直流電源值得轉換&#xff09;&#xff0c;是一種在直流電路中將一個電壓值的電能變為另一個電壓值的電能裝置。 DC-DC轉換器一般由控制芯片、電感線圈、二極管、三極管、電容器構成。 2. 基本拓撲結構 2.1 非隔離…

JavaEE:分布式session

一、使用Redis存儲分布式session&#xff1a; 1.SpringBoot整合Redis&#xff0c;見如下地址&#xff1a; JavaEE&#xff1a;SpringBoot整合Redis_a526001650a-CSDN博客 2.代碼實現分布式session存儲(此處以token為例)&#xff1a; Autowired private RedisTemplate<St…

OpenCV CUDA模塊設備層-----“大于閾值設為零” 的圖像處理函數 thresh_to_zero_inv_func()

操作系統&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 編程語言&#xff1a;C11 算法描述 OpenCV 的 CUDA 模塊&#xff08;cudev&#xff09; 中的一個仿函數生成器&#xff0c;用于創建一個 “大于閾值設為零” 的圖像處理函數對象。 …

FastGPT與MCP:解鎖AI新時代的技術密碼

一、AI 浪潮中的新星&#xff1a;FastGPT 與 MCP 登場 在當今科技飛速發展的時代&#xff0c;人工智能&#xff08;AI&#xff09;已成為推動各行業變革的核心力量。從智能語音助手到復雜的圖像識別系統&#xff0c;AI 的應用無處不在&#xff0c;而其中的關鍵技術 —— 語言模…

browser-tools-mcp + excel-mcp-server + cursor 實現讀取網頁信息自動寫入Excel

browser-tools-mcp excel-mcp-server cursor 實現讀取網頁信息自動寫入Excel 文章目錄 browser-tools-mcp excel-mcp-server cursor 實現讀取網頁信息自動寫入Excel一、安裝node.js和npm1、安裝nvm2、安裝最新版本的node.js 二、安裝browser-tools-mcp1、安裝 BrowserTools…