构造析构函数的妙用-成对出现

类别:编程语言 点击:0 评论:0 推荐:

构造析构的妙用--成对出现

考虑这样一段代码:
void some_func()
{
 lock(key);
 ……
 ……
 unlock(key);
}
由于对some_func的不能并发访问,所以在函数的开始处加上锁key,只有key处

于解锁状态,lock才可以返回,当函数执行完毕后unlock,其他进程调用lock

方能成功,从而可以得以执行some_func。

在函数体内如果出现这样的情况:
void some_func()
{
 lock(key);
 .....
 if(something)
   return;
 .....
 unlock(key);
}
函数某些条件的判断而返回,这样,some_func存在了大于一个返回点,这样由

于没有执行unlock操作,其他进程或本进程对some_func的再次访问将被死锁而

永远无法得以执行。
如果我们这样解决问题:
void some_func()
{
 lock(key);
 .....
 if(something)
 {
  unlock(key);
  return;
 }
 .....
 unlock(key);
}
这样可以解决部分问题,但是如果函数返回点很多,就需要在每个返回点进行

解锁unlock操作,很麻烦,同时它还存在如下问题:
void some_func()
{
 lock(key);
 ....
 my_func();//由其他程序员实现的函数。
 .....
 unlock(key);
}
看起来一点问题也没有,但是如果my_func()丢出异常呢?
如果我们函数里没有捕获这个异常的话 ,它会导致 some_func 函数在调用

my_func的这一点中断。那么只能在每个可能抛出异常的函数调用点用 try 捕

获所有异常,然后在 catch 里面解锁,再重新抛出。显然,这样的解法实在是

太繁琐、太容易引入错误了。

如果我们这样考虑,锁和解锁的操作是要成对出现的,而对于一个类而讲,构

造和析构函数也是成对出现的,我们可以这样设计:
class auto_lock
{
public:
 auto_lock(lock_t key) : m_key(key)
 {
  lock(m_key);// 构造时加锁
 }
 ~auto_lock()
 {
  unlock(m_key)// 析构时解锁
 }

private:
 lock_t m_key;
}

void some_func()
{
 auto_lock my_lock(key);
 ……
 // return 、foo ,随便什么东西都行
 ……
 // 结束的时候同样不用解锁
}
由于my_lock是一个局部变量,他的生命周期在函数some_func内,不论some_fu

nc以什么样的方式退出,my_lock都将被销毁,析构函数被执行,锁被解开,彻

底解决了死锁的问题,解决的很完美。

同样在其他有成对出现的情况下都可以考虑使用构造和析构函数。

本文地址:http://com.8s8s.com/it/it29027.htm