0

0

c++中如何实现一个简单的自动释放锁RAII类_c++构造与析构技巧【详解】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-22 17:36:16

|

600人浏览过

|

来源于php中文网

原创

可手动实现最小RAII锁包装器:构造时加锁、析构时解锁,禁用拷贝,用指针存锁对象以避免临时对象绑定;若需延迟加锁,则分离lock()调用,如simple_unique_lock。

c++中如何实现一个简单的自动释放锁raii类_c++构造与析构技巧【详解】

为什么不能直接用 std::lock_guard 就自己写?

多数场景下,std::lock_guard 已足够——它在构造时加锁、析构时自动释放,符合 RAII 原则。但自己实现一个简化版,有助于理解底层逻辑:比如控制加锁时机(延迟构造时不立即加锁)、支持可重入判断、或封装自定义锁类型(如读写锁的升级逻辑)。关键不是“替代”,而是“可控”。

如何手动实现一个最小可用的 RAII 锁包装器?

核心是把锁对象指针存为成员,并在析构函数中调用 unlock();构造函数是否加锁,取决于设计意图。常见做法是「构造即加锁」,与 std::lock_guard 一致:

template 
class simple_lock_guard {
    Mutex* mtx_;
    bool owns_;

public: explicit simple_lockguard(Mutex& mtx) : mtx(&mtx), owns(true) { mtx.lock(); // 构造时阻塞加锁 }

~simple_lock_guard() {
    if (owns_) mtx_-youjiankuohaophpcnunlock();
}

simple_lock_guard(const simple_lock_guard&) = delete;
simple_lock_guard& operator=(const simple_lock_guard&) = delete;

};

注意点:

立即学习C++免费学习笔记(深入)”;

  • mtx_ 存指针而非引用,避免临时对象绑定问题
  • 禁用拷贝——RAII 对象不可复制,否则析构时会重复 unlock
  • 未实现移动语义,若需转移所有权,得额外加 release() 方法并置 owns_ = false

构造时不加锁,到需要时再 lock 怎么办?

这是 std::unique_lock 的典型行为,适用于条件变量等待、延迟加锁等场景。手动实现只需分离构造与加锁动作:

灵光
灵光

蚂蚁集团推出的全模态AI助手

下载
template 
class simple_unique_lock {
    Mutex* mtx_;
    bool owns_;

public: simple_uniquelock() : mtx(nullptr), owns_(false) {}

explicit simple_unique_lock(Mutex& mtx) : mtx_(&mtx), owns_(false) {}

void lock() {
    if (!owns_ && mtx_) {
        mtx_-youjiankuohaophpcnlock();
        owns_ = true;
    }
}

void unlock() {
    if (owns_ && mtx_) {
        mtx_-youjiankuohaophpcnunlock();
        owns_ = false;
    }
}

~simple_unique_lock() {
    if (owns_ && mtx_) mtx_-youjiankuohaophpcnunlock();
}

// 禁用拷贝,允许移动(可选)
simple_unique_lock(const simple_unique_lock&) = delete;
simple_unique_lock& operator=(const simple_unique_lock&) = delete;

};

常见误用:

  • 忘记检查 owns_ 就调用 unlock() → 可能对未加锁的 mutex 解锁,UB
  • 构造传入临时 Mutex 对象 → mtx_ 指向已销毁内存
  • 移动后未重置原对象的 owns_mtx_ → 析构时 double-unlock

析构函数里抛异常会怎样?

如果 unlock() 抛异常(比如锁已被破坏、系统资源失效),而此时正在展开(例如另一个异常已发生),程序会直接调用 std::terminate。C++ 标准要求互斥量的 unlock() 是 noexcept 的,所以实际中应确保:

  • 所用 Mutex 类型的 unlock() 不抛异常(如 std::mutex 满足)
  • 不要在 RAII 析构函数里做任何可能抛异常的操作(日志、网络调用等)
  • 若必须处理错误,用 try/catch 吞掉异常并记录,但不 re-throw

真正容易被忽略的是:析构函数的异常安全性不是靠“写得漂亮”,而是靠“别让它发生”。RAII 的干净,建立在底层资源操作的确定性之上。

相关专题

更多
c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

53

2025.08.29

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

100

2025.10.23

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

393

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

574

2023.08.10

Golang 性能分析与pprof调优实战
Golang 性能分析与pprof调优实战

本专题系统讲解 Golang 应用的性能分析与调优方法,重点覆盖 pprof 的使用方式,包括 CPU、内存、阻塞与 goroutine 分析,火焰图解读,常见性能瓶颈定位思路,以及在真实项目中进行针对性优化的实践技巧。通过案例讲解,帮助开发者掌握 用数据驱动的方式持续提升 Go 程序性能与稳定性。

9

2026.01.22

html编辑相关教程合集
html编辑相关教程合集

本专题整合了html编辑相关教程合集,阅读专题下面的文章了解更多详细内容。

56

2026.01.21

三角洲入口地址合集
三角洲入口地址合集

本专题整合了三角洲入口地址合集,阅读专题下面的文章了解更多详细内容。

30

2026.01.21

AO3中文版入口地址大全
AO3中文版入口地址大全

本专题整合了AO3中文版入口地址大全,阅读专题下面的的文章了解更多详细内容。

393

2026.01.21

妖精漫画入口地址合集
妖精漫画入口地址合集

本专题整合了妖精漫画入口地址合集,阅读专题下面的文章了解更多详细内容。

116

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

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