C++ 互斥锁的使用

news2025/2/23 23:41:16

mutex

std::mutex 是C++标准库中用于线程同步的互斥锁机制,主要用于保护共享资源,避免多个线程同时访问导致的竞态条件。

它提供了以下功能:

  • 加锁(lock:阻塞当前线程,直到获取锁。

  • 解锁(unlock:释放锁,允许其他线程获取锁。

  • 尝试加锁(try_lock:尝试获取锁,如果锁已被占用则立即返回。

使用全局锁案例

#include <iostream>
#include <thread>
#include <mutex>
using namespace std;
mutex mtx; // 定义一个全局互斥锁
int shared_data = 0;

void increment() {
    for (int i = 0; i < 10000; i++) {
        mtx.lock(); // 加锁
        shared_data++;
        mtx.unlock(); // 解锁
    }
}

int main() {
    thread t1(increment);
    thread t2(increment);
    t1.join();
    t2.join();

    cout << "共享数据: " << shared_data << endl; // 输出应该是20000
    return 0;
}

使用传参锁案例

如果线程对象传参给可调⽤对象时,使⽤引⽤⽅式传参,实参位置需要加上ref(obj)的⽅式,主要原因是thread本质还是系统库提供的线程API的封装,thread 构造取到参数包以后,要调⽤创建线程的API,还是需要将参数包打包成⼀个结构体传参过去,那么打包成结构体时,参考包对象就会拷⻉给结构体对象,使⽤ref传参的参数,会让结构体中的对应参数成员类型推导为引⽤,这样才能实现引⽤传参。
#include <iostream>
#include <thread>
#include <mutex>
using namespace std;
int shared_data = 0;

void increment(mutex& mtx) {
    for (int i = 0; i < 10000; i++) {
        mtx.lock(); // 加锁
        shared_data++;
        mtx.unlock(); // 解锁
    }
}

int main() {
    mutex mtx; // 使用传参互斥锁
    thread t1(increment,ref(mtx));//需要使用 ref 包装锁,不然会报错
    thread t2(increment,ref(mtx));
    t1.join();
    t2.join();

    cout << "共享数据: " << shared_data << endl; // 输出应该是20000
    return 0;
}

timed_mutex

std::timed_mutex 是C++11引入的一种互斥锁类型,用于多线程编程中控制对共享资源的并发访问。它提供了超时机制,允许线程在尝试获取锁时设置一个时间限制,从而避免无限等待锁释放,降低死锁风险。

timed_mutex 跟 mutex完全类似,只是额外提供 try_lock_for 和 try_lock_untile 的接⼝,这两个接⼝跟 try_lock 类似,只是他不会⻢上返回,⽽是直接进⼊阻塞,直到时间条件到了或者解锁了就会唤醒试图获取锁资源。
  • try_lock_for(duration):尝试在指定的时间内获取锁,如果超时则返回 false

  • try_lock_until(time_point):尝试在指定的时间点之前获取锁,如果超时则返回 false。

#include <iostream>
#include <thread>
#include <mutex>
#include <chrono>
using namespace std;
timed_mutex mtx;

void func(int id) {
    chrono::seconds timeout(3); // 设置超时时间为3秒
    if (mtx.try_lock_for(timeout)) {
        cout << "线程 " << id << " 成功获取锁,执行临界区代码..." << endl;
        this_thread::sleep_for(chrono::seconds(5)); // 模拟工作负载
        mtx.unlock();
    }
    else {
        cout << "线程 " << id << " 超时未能获取锁,继续执行其他任务..." << endl;
    }
}

int main() {
    thread t1(func, 1);
    thread t2(func, 2);
    t1.join();
    t2.join();

    return 0;
}

recursive_mutex

recursive_mutex 跟 mutex 完全类似,允许同一个线程多次获取锁而不会导致死锁,这种互斥锁特别适用于递归函数或嵌套调用的场景。 recursive_mutex 提供排他性递归所有权语义:
  1. 调⽤⽅线程在从它成功调⽤ lock 或 try_lock 开始的时期⾥占有 recursive_mutex。此时期之内,线程可以进⾏对 lock 或 try_lock 的附加调⽤。所有权的时期在线程进⾏匹配次数的 unlock 调⽤时结束。
  2. 线程占有 recursive_mutex 时,若其他所有线程试图要求 recursive_mutex 的所有权,则它们将阻塞(对于调⽤ lock)或收到 false 返回值(对于调⽤ try_lock)。
#include <iostream>
#include <mutex>
#include <thread>
using namespace std;
recursive_mutex rec_mutex;

void recursiveFunction(int count) {
    if (count <= 0) return;
    rec_mutex.lock();
    cout << "Lock acquired. Count: " << count << endl;
    // 递归调用
    recursiveFunction(count - 1);

    rec_mutex.unlock();
    cout << "Lock released. Count: " << count << endl;
}

int main() {
    thread t(recursiveFunction, 3);
    t.join();
    return 0;
}

lock_guard

std::lock_guard 是 C++ 标准库中用于管理互斥锁的 RAII(Resource Acquisition Is Initialization,资源获取即初始化)工具。它通过构造函数自动获取互斥锁,并在析构函数中自动释放锁,从而确保互斥锁的正确管理,避免因忘记解锁而导致的死锁问题。

主要特性

  1. 自动管理锁:通过构造函数获取锁,析构函数释放锁,无需手动调用 lock()unlock()

  2. 防止死锁:确保即使在异常情况下也能正确释放锁。

  3. 不可移动和复制:std::lock_guard 不支持拷贝或移动构造,确保锁的唯一性。

构造函数

explicit lock_guard(MutexType& mutex, std::adopt_lock_t tag = std::defer_lock);
  • MutexType& mutex:要管理的互斥锁对象,如mutex,timed_mutex,recersive_mutex。

  • std::adopt_lock_t tag:可选参数,表示当前线程已经持有锁,默认值是 std::defer_lock,表示 lock_guard 会在构造时自动尝试加锁。如果传入 std::adopt_lock,则表示当前线程已经通过其他方式(例如 std::lock 或手动调用 lock())持有锁,lock_guard 不会再次尝试加锁,而是直接接管锁的管理。

#include <iostream>
#include <thread>
#include <mutex>
using namespace std;
mutex mtx;
void print_block(int n, char c) {
    lock_guard<mutex> lock(mtx);  // 默认加锁,无需手动解锁
    for (int i = 0; i < n; ++i) {
        cout << c;
    }
    cout << '\n';
}

int main() {
    thread t1(print_block, 50, '*');
    thread t2(print_block, 50, '$');
    t1.join();
    t2.join();
    return 0;
}

unique_lock

unique_lock也是C++11提供的⽀持RAII⽅式管理互斥锁资源的类,相⽐ lock_guard 他的功能⽀持
更丰富复杂。

主要特性

  1. 延迟加锁:可以在构造时不立即加锁,而是通过显式调用 lock()unlock() 来控制锁的获取和释放。

  2. 支持条件变量:可以与 std::condition_variable 配合使用,支持线程间的同步。

  3. 可选锁管理:可以选择是否持有锁,或者在构造时直接接管已持有的锁。

  4. 支持多种互斥锁类型:可以与 std::mutexstd::recursive_mutexstd::timed_mutexstd::shared_mutex 配合使用。

构造函数

template <typename Mutex>
unique_lock() noexcept;  // 默认构造,不绑定任何锁

template <typename Mutex>
explicit unique_lock(Mutex& m, std::defer_lock_t);  // 延迟加锁

template <typename Mutex>
unique_lock(Mutex& m, std::try_to_lock_t);  // 尝试加锁,失败时不阻塞

template <typename Mutex>
unique_lock(Mutex& m, std::adopt_lock_t);  // 假设已持有锁,不加锁

template <typename Mutex>
unique_lock(Mutex& m);  // 默认加锁

template <typename Mutex>
unique_lock(Mutex& m, std::unique_lock<Mutex>&& other);  // 移动构造

主要成员函数

  1. lock():尝试加锁,如果锁已被占用则阻塞。

  2. try_lock():尝试加锁,如果锁已被占用则立即返回 false

  3. unlock():释放锁。

  4. owns_lock():检查是否持有锁。

  5. mutex():获取绑定的互斥锁对象。

  6. release():释放锁的所有权,但不释放锁本身。

  7. swap():交换两个 std::unique_lock 的状态。

lock和try_lock

lock是⼀个函数模板,可以⽀持对多个锁对象同时锁定,如果其中⼀个锁对象没有锁住,lock函数
会把已经锁定的对象解锁⽽进⼊阻塞,直到锁定所有的所有的对象,它的主要目的是避免死锁问题,通过一次性尝试锁定多个锁,确保所有锁都被成功锁定后才返回。
try_lock也是⼀个函数模板,尝试对多个锁对象进⾏同时尝试锁定,如果全部锁对象都锁定了,返 回-1,如果某⼀个锁对象尝试锁定失败,把已经锁定成功的锁对象解锁,并则返回这个对象的下标
(第⼀个参数对象,下标从0开始算)。
#include <iostream>
#include <thread>
#include <mutex>
using namespace std;
mutex mtx1;
mutex mtx2;
void worker() {
    lock(mtx1, mtx2);  // 同时锁定两个锁
    unique_lock<mutex> lock1(mtx1, std::adopt_lock);  // 需要创建 unique_lock 自动释放锁
    unique_lock<mutex> lock2(mtx2, std::adopt_lock);

    cout << "线程进入临界区..." << endl;
    this_thread::sleep_for(chrono::seconds(1));  // 模拟工作负载
    cout << "线程退出临界区..." << endl;
}

int main() {
    thread t1(worker);
    thread t2(worker);
    t1.join();
    t2.join();
    return 0;
}

call_once

std::call_once 是 C++11 引入的一个函数模板,用于确保某个操作(如初始化或配置加载)在多线程环境中只被调用一次。它结合了 std::once_flag,能够高效地实现线程安全的单次执行。

基本用法

std::call_once 的函数原型如下:

template <class Callable, class... Args>
void call_once(std::once_flag& flag, Callable&& f, Args&&... args);
  • std::once_flag:一个标记对象,用于记录操作是否已经执行过。

  • Callable&& f:需要执行的可调用对象(如函数、lambda 表达式、函数对象等)。

  • Args&&... args:传递给可调用对象的参数。

工作原理

  1. 首次调用:如果 std::once_flag 标记的操作尚未执行,则 std::call_once 会调用 f,并设置 flag 标记为“已执行”。

  2. 后续调用:如果 flag 已标记为“已执行”,则后续调用 std::call_once 的线程会直接跳过 f 的执行。

#include <iostream>
#include <thread>
#include <mutex>
using namespace std;

std::once_flag flag;
void init(int id) {
    cout << "Initialization function called once." << endl;
    cout << "thread " << id << " is running." << endl;
}

void worker(int id) {
    call_once(flag, init,id);  // 确保 init 只被调用一次
}

int main() {
    thread t1(worker, 1);
    thread t2(worker, 2);
    thread t3(worker, 3);
    t1.join();
    t2.join();
    t3.join();
    return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2304126.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

车载诊断架构 --- LIN节点路由转发注意事项

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 简单,单纯,喜欢独处,独来独往,不易合同频过着接地气的生活,除了生存温饱问题之外,没有什么过多的欲望,表面看起来很高冷,内心热情,如果你身…

Eclipse2024中文汉化教程(图文版)

对应Eclipse,部分人需要中文汉化,本章教程,介绍如何对Eclipse进行汉化的具体步骤。 一、汉化前的Eclipse 默认安装Eclipse的时候,默认一般都是English的,我当前版本是使用的是2024-06版本的Eclipse。 二、汉化详细步骤 点击上方菜单选项卡,Hep——Install New Software……

医院安全(不良)事件上报系统源码,基于Laravel8开发,依托其优雅的语法与强大的扩展能力

医院安全&#xff08;不良&#xff09;事件上报系统源码 系统定义&#xff1a; 规范医院安全&#xff08;不良&#xff09;事件的主动报告&#xff0c;增强风险防范意识&#xff0c;及时发现医院不良事件和安全隐患&#xff0c;将获取的医院安全信息进行分析反馈&#xff0c;…

【第一节】C++设计模式(创建型模式)-工厂模式

目录 前言 一、面向对象的两类对象创建问题 二、解决问题 三、工厂模式代码示例 四、工厂模式的核心功能 五、工厂模式的应用场景 六、工厂模式的实现与结构 七、工厂模式的优缺点 八、工厂模式的扩展与优化 九、总结 前言 在面向对象系统设计中&#xff0c;开发者常…

爬虫小案例豆瓣电影top250(json格式)

1.json格式&#xff08;仅供学习参考&#xff09; import requests, json, jsonpathclass Start(object):# 类实例化时会执行def __init__(self):self.headers {user-agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/107.0.…

Spring事务原理 二

在上一篇博文《Spring事务原理 一》中&#xff0c;我们熟悉了Spring声明式事务的AOP原理&#xff0c;以及事务执行的大体流程。 本文中&#xff0c;介绍了Spring事务的核心组件、传播行为的源码实现。下一篇中&#xff0c;我们将结合案例&#xff0c;来讲解实战中有关事务的易…

SpringAI系列 - ToolCalling篇(二) - 如何设置应用侧工具参数ToolContext(有坑)

目录 一、引言二、集成ToolContext示例步骤1: 在`@Tool`标注的工具方法中集成`ToolConext`参数步骤2:`ChatClient`运行时动态设置`ToolContext`参数三、填坑一、引言 在使用AI大模型的工具调用机制时,工具参数都是由大模型解析用户输入上下文获取的,由大模型提供参数给本地…

本地部署MindSearch(开源 AI 搜索引擎框架),然后上传到 hugging face的Spaces——L2G6

部署MindSearch到 hugging face Spaces上——L2G6 任务1 在 官方的MindSearch页面 复制Spaces应用到自己的Spaces下&#xff0c;Space 名称中需要包含 MindSearch 关键词&#xff0c;请在必要的步骤以及成功的对话测试结果当中 实现过程如下&#xff1a; 2.1 MindSearch 简…

MyBatis Plus扩展功能

一、代码生成器 二、逻辑删除 三、枚举处理器 像状态字段我们一般会定义一个枚举&#xff0c;做业务判断的时候就可以直接基于枚举做比较。但是我们数据库采用的是int类型&#xff0c;对应的PO也是Integer。因此业务操作时必须手动把枚举与Integer转换&#xff0c;非常麻烦。 …

深度学习之自然语言处理CBOW预测及模型的保存

自然语言处理CBOW预测及模型的保存 目录 自然语言处理CBOW预测及模型的保存1 自然语言处理1.1 概念1.2 词向量1.2.1 one-hot编码1.2.2 词嵌入1.2.3 常见的词嵌入模型 2 CBOW预测模型搭建2.1 数据及模型确定2.1.1 数据2.1.2 CBOW模型2.1.3 词嵌入降维 2.2 数据预处理2.3 模型搭建…

qt项目配置部署

Test项目: 子项目testFileHelper 1.新建一个test项目的子项目:取名testFileHelper 2.编写测试用例 3.pro文件中引入qosbrowser 4.引入测试对象的cpp和头文件 2.在项目中引入资源文件testfile.txt,在其中输入abc 实现thrid目录复用 移动thrid 将thrild目录统一放在章…

java方法学习

java 方法 在Java中&#xff0c;方法是类&#xff08;或对象&#xff09;的行为或功能的实现。&#xff08;一起实现一个功能&#xff09;java的方法类似于其他语言的函数&#xff0c;是一段用来完成特定功能的代码片段。 方法是解决一类问题步骤的有序结合。 方法包含于类或…

基于vue和微信小程序的校园自助打印系统(springboot论文源码调试讲解)

第3章 系统设计 3.1系统功能结构设计 本系统的结构分为管理员和用户、店长。本系统的功能结构图如下图3.1所示&#xff1a; 图3.1系统功能结构图 3.2数据库设计 本系统为小程序类的预约平台&#xff0c;所以对信息的安全和稳定要求非常高。为了解决本问题&#xff0c;采用前端…

[漏洞篇]文件上传漏洞详解

[漏洞篇]文件上传漏洞详解 一、介绍 1. 概念 文件上传漏洞是指用户上传了一个可执行的脚本文件&#xff0c;并通过此脚本文件获得了执行服务器端命令的能力。这种攻击方式是最为直接和有效的&#xff0c;“文件上传” 本身没有问题&#xff0c;有问题的是文件上传后&#xf…

11.Docker 之分布式仓库 Harbor

Docker 之分布式仓库 Harbor Docker 之分布式仓库 Harbor1. Harbor 组成2. 安装 Harbor Docker 之分布式仓库 Harbor Harbor 是一个用于存储和分发 Docker 镜像的企业级 Registry 服务器&#xff0c;由 VMware 开源&#xff0c;其通过添加一些企业必需的功能特性&#xff0c;例…

Python项目源码34:网页内容提取工具1.0(Tkinter+requests+html2text)

------★Python练手项目源码★------- Python项目32&#xff1a;订单销售额管理系统1.0&#xff08;TkinterCSV&#xff09; Python项目31&#xff1a;初学者也能看懂的聊天机器人1.0源码&#xff08;命令行界面Re正则表达式&#xff09; Python项目源码30&#xff1a;待办事…

使用Termux将安卓手机变成随身AI服务器(page assist连接)

通过以下方法在安卓手机上运行 Ollama 及大模型&#xff0c;无需 Root 权限&#xff0c;具体方案如下&#xff1a; 通过 Termux 模拟 Linux 环境运行 核心工具&#xff1a; 安装 &#xff08;安卓终端模拟器&#xff09;()]。借助 proot-distro 工具安装 Linux 发行版&#xf…

flink-cdc同步数据到doris中

1 创建数据库和表 1.1 数据库脚本 这样直接创建数据库是有问题&#xff0c;因为后面发现superset连接使用doris://root:12345610.101.12.82:9030/internal.eayc?charsetutf8mb4 -- 创建数据库eayc create database if not exists ods_eayc; -- 创建数据表2 数据同步 2.1 f…

Git命令行入门

诸神缄默不语-个人CSDN博文目录 之前写过一篇VSCode Git的博文&#xff1a;VSCode上的Git使用手记&#xff08;持续更新ing…&#xff09; 现在随着开发经历增加&#xff0c;感觉用到命令行之类复杂功能的机会越来越多了&#xff0c;所以我专门再写一篇Git命令行的文章。 G…

DeepSeek R1/V3满血版——在线体验与API调用

前言&#xff1a;在人工智能的大模型发展进程中&#xff0c;每一次新模型的亮相都宛如一颗投入湖面的石子&#xff0c;激起层层波澜。如今&#xff0c;DeepSeek R1/V3 满血版强势登场&#xff0c;为大模型应用领域带来了全新的活力与变革。 本文不但介绍在线体验 DeepSeek R1/…