c++多线程并发(线程互锁,死锁,设计模式等)

c++多线程并发

互斥量:Mutex类

lock unlock 成对使用,Lock是阻塞式的,有一个地方lock, 第二个地方的lock会一直卡在lock里,直到lock成功;

Lock_guard类模版(推荐使用),直接取代lock和unlock;Lock_guard的构造函数里执行了lock,Lock_guard的析构函数里执行了unlock;局部变量超出作用域名时,生命周期结束并 执行析构函数;

加 { } 可以提前结束里面的局部变量,提前执行循环里所定义局部变量的析构函数;

锁住的代码的多少称为锁的粒度,粒度一般用粗细来描述,锁住的越少,执行的效率越高;

死锁:两个线程争夺资源的过程中,两个互斥量嵌套使用,且使用的顺序不一致;

产生条件:必须有两个互斥量(这里比作金锁,银锁),保护不同的数据共享块;

线程A和B,线程A先锁金锁,后锁银锁;线程B先锁银锁,后锁金锁;

当线程A,金锁lock 成功后,开始进行银锁的lock,这时候,线程后台切换到了线程B,导致银锁lock , 线程B开始执行金锁lock,卡在了金锁lock,这时候线程管理又切换到线程A,线程A开在了银锁lock,这样就造成了死锁,两个线程都卡在了lock函数里,无法解开;

死锁的一般解决方案:

保证互斥量的调用顺序一致;

std::lock函数模版:一次锁住两个或者两个以上的互斥量,要么锁住两个,要么两个都没锁住 ,只锁住一个的时候会主动放开,不会造成死锁问题;(至少两个)

Unique_lock取代Lock_guard:

第二个参数(和Lock_guard一样):adopt_lock 起到标记作用,标记互斥量已经锁过了,不需要构造函数中再lock;

try_to_lock:不能之前lock,尝试用mutex的lock去锁定,但如果没有锁定成功,我也会立即返回,不会阻塞在哪里等待,没有拿到锁可以干其他的事情,不会使线程阻塞在那里傻等;

defer_lock:不能之前lock,初始化了一个没加锁的mutex;

成员:lock,unlock,try_lock(();

release(): 返回所管理的mutex对象指针,并释放所有权;

所有权传递问题:

Unique_lock是在管理mutex指针对象,只和一个mutex绑定到一起,拥有mutex的所有权,可以讲所有权转移给其她的unique_lock,可以移动所有权但不能复制,

单例设计模式:

单例类:整个项目中,有某个或者某些特殊的类,属于该类的对象,我只能创建一个,多了创建不了;

方法:把构造函数私有化;

class mycas{  //单例类
private:
				mycas() {}//私有化了构造函数 
private:
  static mycas *m_Instance;//静态成员变量
public:
  static mycas *getInstance()
  {
    if(m_Instance == NULL )//双重锁定,解决多个线程创建单例类时的临界不安全问题,提高效率
    {
      std::unique_lock<std::mutex>mymutex(resource_mutex);//自动加锁
      if(m_Instance == NULL )
      {
        m_Instance = new mycas();
        static cGarhuishou cl; //为了释放m_Instance所指的内存空间
      } 
    }
    else  
      return  m_Instance;
  }
  class cGarhuishou //类中套类,用来释放new的mycas
  {
    public:
    ~cGarhuishou() //类的析构函数
    {
       if(mycas::m_Instance)//如果类对象指针不为空
       {
         delete mycas::m_Instance;
         mycas::m_Instance = NULL;
       }     
    }
  };
};
//类静态变量初始化
mycas *mycas::m_Instance = NULL;
int main()
{
  mycas *p_A = mycas::getInstance();//创建一个单例类对象,且只能创建一次
  mycas *p_B = mycas::getInstance();//其实是与p_A指向的空间相同的;
  // mycas p_A  的方式是无法创建的;
}

单例设计模式共享数据问题分析与解决

在自己或者其他线程中需要创建使用单例类对象,我们可能会面临getInstance()这种成员函数要互斥;

两个线程都要创建单例类等时候,存在调用getInstance()时判断条件由于线程自动切换产生歧义,导致创建了两个单例类,这个时候采用互斥量可以解决;

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值