自动锁

新学了一个技巧:自动锁,类似于ace的自动锁,进入作用域声明一个用mutex初始化的自动锁对象,对象声明成功则表示获取锁资源成功,程序继续运行,否则持续等待;在离开作用域时自动释放锁。

class AutoLock
{
public:
    AutoLock(pthread_mutex_t* pMutex):m_mutex(pMutex)
    {
        pthread_mutex_lock(m_mutex);
    }
    ~AutoLock()
    {
        pthread_mutex_unlock(m_mutex);
    }
private:
    pthread_mutex_t *m_mutex;
};

调用则如下 


//预处理,初始化锁资源
pthread_mutexattr_t attr;
pthread_mutexattr_init(&attr);
pthread_mutexattr_setpshared(&attr, PTHREAD_PROCESS_SHARED);

pthread_mutex_t tid;        
pthread_mutex_init(&tid, &attr);

function()
{
    //进入作用域自动锁定资源,且离开作用域自动释放资源,不需要每一个出口手动释放资源,提高健壮性
    DCPParamAutoLock   lock(&tid);
    //do something and leave effect zone
}

     嗯,简单好用。

     补充:

     简单版本的自动锁随作用域而生,无法解决多分支的加锁和解锁要求,也无法处理递归调用方法的重复加锁,可以给成员增加变量标志是否加锁,同时增加acquire和release接口,避免重复加锁和支持分支加解锁。

     加锁要考虑的问题有:多分支退出,需要保证所有分支均解锁;异常退出,要保证退出作用域时解锁;递归调用,每次递归退出要对应解锁。

     多线程并发编程应该规避无法正常解锁的场景:线程在处理过程中异常应该抛出异常并退出,不应该直接退出线程,直接退出线程不会调用当前堆栈上对象的析构函数,会导致其他活动线程无法获得临界区资源。