首页 > 后端开发 > C++ > 正文

c++怎么使用std::shared_mutex实现读写锁_c++读写锁shared_mutex用法详解

裘德小鎮的故事
发布: 2025-10-29 16:37:02
原创
141人浏览过
std::shared_mutex是C++17引入的读写锁机制,允许多个线程共享读锁、单个线程独占写锁,适用于读多写少场景,需包含<shared_mutex>头文件并启用C++17,推荐使用std::shared_lock和std::unique_lock进行RAII管理以确保异常安全和防止死锁。

c++怎么使用std::shared_mutex实现读写锁_c++读写锁shared_mutex用法详解

std::shared_mutex 是 C++17 引入的一个同步原语,用于实现高效的读写锁机制。它允许多个线程同时进行读操作(共享访问),但在写操作(独占访问)时只允许一个线程执行,并且会阻塞所有其他读和写线程。这种机制非常适合“读多写少”的场景,比如缓存、配置管理等。

1. std::shared_mutex 基本概念

std::shared_mutex 区分两种锁定模式:
  • 共享锁(shared lock):多个线程可以同时持有,适用于读操作。
  • 独占锁(exclusive lock):仅一个线程能持有,适用于写操作。
当有线程持有共享锁时,新的共享锁可以立即获得;但独占锁必须等待所有共享锁释放后才能获取。

2. 头文件与编译要求

使用 std::shared_mutex 需要包含头文件:

#include <shared_mutex>

火龙果写作
火龙果写作

用火龙果,轻松写作,通过校对、改写、扩展等功能实现高质量内容生产。

火龙果写作106
查看详情 火龙果写作
并且编译器需支持 C++17 或更高版本。例如用 g++ 编译时加上:

g++ -std=c++17 your_file.cpp

3. 使用方法示例

下面是一个典型的读写锁使用示例:

#include <iostream>
#include <thread>
#include <vector>
#include <shared_mutex>
#include <chrono>

std::vector<int> data = {1, 2, 3};
std::shared_mutex shm;  // 共享互斥量

// 读操作函数
void reader(int id) {
    shm.lock_shared();  // 获取共享锁
    std::cout << "Reader " << id << " sees data: ";
    for (int d : data) {
        std::cout << d << " ";
    }
    std::cout << "\n";
    std::this_thread::sleep_for(std::chrono::milliseconds(100)); // 模拟处理时间
    shm.unlock_shared();  // 释放共享锁
}

// 写操作函数
void writer(int id) {
    shm.lock();  // 获取独占锁
    std::cout << "Writer " << id << " is writing...\n";
    data.push_back(id);
    std::this_thread::sleep_for(std::chrono::milliseconds(200));
    std::cout << "Writer " << id << " finished writing.\n";
    shm.unlock();  // 释放独占锁
}
登录后复制

主函数中启动多个读线程和写线程:

int main() {
    std::vector<std::thread> threads;

    // 启动 5 个读线程
    for (int i = 1; i <= 5; ++i) {
        threads.emplace_back(reader, i);
    }

    // 启动 2 个写线程
    for (int i = 10; i <= 11; ++i) {
        threads.emplace_back(writer, i);
    }

    // 启动更多读线程
    for (int i = 6; i <= 10; ++i) {
        threads.emplace_back(reader, i);
    }

    // 等待所有线程完成
    for (auto& t : threads) {
        t.join();
    }

    return 0;
}
登录后复制

输出可能类似(顺序不固定):

Reader 1 sees data: 1 2 3 Reader 2 sees data: 1 2 3 ... Writer 10 is writing... Writer 10 finished writing. Reader 6 sees data: 1 2 3 10 ...

4. 推荐使用 RAII 管理锁

直接调用 lock/unlock 容易出错,建议使用 RAII 封装类自动管理锁的生命周期:
  • std::shared_lock<std::shared_mutex>:用于管理共享锁。
  • std::unique_lock<std::shared_mutex>:也可用于独占锁,但更常用的是直接用 lock_guard 或 unique_lock 配合普通 mutex。不过 shared_mutex 支持 unique_lock 的独占模式。
改写上面的函数更安全的方式:

void reader(int id) {
    std::shared_lock<std::shared_mutex> lock(shm);  // 自动加共享锁,析构时释放
    std::cout << "Reader " << id << " sees data: ";
    for (int d : data) {
        std::cout << d << " ";
    }
    std::cout << "\n";
    std::this_thread::sleep_for(std::chrono::milliseconds(100));
}  // lock 在这里自动释放

void writer(int id) {
    std::unique_lock<std::shared_mutex> lock(shm);  // 独占锁
    std::cout << "Writer " << id << " is writing...\n";
    data.push_back(id);
    std::this_thread::sleep_for(std::chrono::milliseconds(200));
    std::cout << "Writer " << id << " finished writing.\n";
}  // lock 析构时自动释放
登录后复制

这种方式避免了忘记 unlock 导致死锁的风险,也支持异常安全。

5. 注意事项与最佳实践

使用 std::shared_mutex 时注意以下几点:
  • 确保读写操作逻辑正确区分:只读用 shared_lock,修改用 unique_lock 或 lock()。
  • 避免长时间持有共享锁,否则会阻塞写线程,造成写饥饿。
  • 某些平台实现中,shared_mutex 性能开销比普通 mutex 高,应根据实际场景选择是否使用。
  • 不能递归加锁:同一个线程重复对 shared_mutex 加共享或独占锁会导致未定义行为。
基本上就这些。合理使用 std::shared_mutex 能有效提升多线程程序在读密集场景下的并发性能。

以上就是c++++怎么使用std::shared_mutex实现读写锁_c++读写锁shared_mutex用法详解的详细内容,更多请关注php中文网其它相关文章!

c++速学教程(入门到精通)
c++速学教程(入门到精通)

c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习
PHP中文网抖音号
发现有趣的

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号