zoukankan      html  css  js  c++  java
  • Linux组件封装之四:RAII实现MutexLock自动化解锁

    在实现线程的过程中,我们经常会写类似于这样的代码:

    {
        mutex_.lock();
        //XXX
        if(...)
                语句;
        //XXX
        mutex_.unlock();
    }

    虽然这段代码是正常的加锁解锁,但是有时候我们难免会出现一些低级错误,例如把 忘了写mutex_.unlock()。那么我们该如何防止这种错误呢? 我们可以采用和实现智能指针相似的办法,把加锁和解锁封装在同一个对象中。

    实现“对象生命期”等于“加锁、解锁周期” 。

    代码如下;

     1 //类的关联
     2 class MutexGuard:NonCopyable
     3 {
     4     public:
     5         MutexGuard(MutexLock &mutex)//no copy no assignment
     6             :mutex_(mutex)
     7         { mutex_.lock();  }
     8         
     9         ~MutexGuard()
    10         { mutex_.unlock(); } 
    11    
    12     private:
    13         MutexLock &mutex_;//attention ,这里M类不存在copy,assignment能力故引用之   
    14 };

    这样 我们就把资源的获取(加锁)放在构造函数,把资源的释放(解锁)放在析构函数中,这种做法就是C++中的RAII技术。

    这里我们给出RAII的解释:RAII,也称为“资源获取就是初始化”,是c++等编程语言常用的管理资源、避免内存泄露的方法。它保证在任何情况下,使用对象时先构造对象,最后析构对象
    这对于我们编写简洁优雅的代码,好处是显而易见的。

    以下这个程序实现的是计算Buffer是否为空,一般我们会这样写:

    1 bool Buffer::isEmpty()const
    2 {
    3     bool flag = false;
    4     mutex_lock();
    5     flag = q_.empty();
    6     mutex_.unlock();
    7     return flag;
    8 }

    这段代码实在称不上美观,但是我们使用MutexGuard之后,我们的代码就变为:

    bool Buffer::isEmpty()const
    {
    //after    
        MutexGuard lock(mutex_);//作用域仅限于花括号内,随后自动调用析构函数,放锁
        return q_.empty();
    }

    这是否美观了许多?
    当然,我们有时候会犯这种错误,忘记声明MutexGuard的对象,例如:

    bool Buffer::isEmpty()const
    {
        MutexGuard mutex_);//wrong
        return q_.empty();
    }

    为了防止这种错误,我们增加了一个

    #define MutexGuard(m) "Error MutexGuard"

    这样,当我们错误使用的时候,就会导致编译错误,可以帮助我们早些发现问题。

  • 相关阅读:
    Linux 编程笔记(四)
    渗透测试搜索指令整理(一)
    Linux 编程笔记(三)
    Winhex数据恢复笔记(五)
    Winhex数据恢复学习笔记(四)
    Winhex数据恢复学习笔记(三)
    Linux编程学习笔记(二)
    Linux编程学习笔记(一)
    工控系统安全问题汇总(一)
    WinHex数据恢复笔记(二)
  • 原文地址:https://www.cnblogs.com/xfxu/p/4010350.html
Copyright © 2011-2022 走看看