单例模式 - 双锁机制

目录

1 饿汉模式

2 懒汉模式

3 多线程安全的懒汉单例模式

3.1 普通加锁方式

3.2 双锁机制


1 饿汉模式

在程序启动或单件模式类被加载的时候,单件模式实例就已经被创建。

主要注意如下四个部分:

  • 构造函数私有化或者保护化
  • .禁止拷贝和赋值
  • 静态的共有接口
  • 初始化静态数据成员

懒汉模式

当程序第一次访问单件模式实例时才进行创建。

       懒汉模式下,在定义m_instance变量时先等于NULL,在调用GetInstance()方法时,在判断是否要赋值。这种模式,并非是线程安全的,因为多个线程同时调用GetInstance()方法,就可能导致有产生多个实例。要实现线程安全,就必须加锁

多线程安全的懒汉单例模式

3.1 普通加锁方式

      这种GetInstance()方法,每次进来都要加锁,会影响效率。然而这并不是必须的,于是又对GetInstance()方法进行改进。

3.2 双锁机制

        上述的方式可能会有意料不到的后果。问题的来源是CPU的乱序执行,C++的New操作实际上包含了两个步骤:(1)分配内存;(2)调用构造函数

实际上,m_instance = new T()包含了三个步骤:

  1. 分配内存
  2. 在内存的位置上调用构造函数
  3. 将内存的地址赋值给pInst

          因为(2)和(3)是可以颠倒的,所以可以出现这样的情况:m_instance的值已经不是NULL,但对象仍然没有构造完毕。如果另外一个线程对GetInstance的调用,此时第一个if为false,这样就会返回一个未构造完成的对象,此时可能会导致程序崩溃
 

解决思路:用一个局部变量过渡下就ok了

       在linux提供了一个叫pthread_once()的函数,它保证在一个进程中,某个函数只被执行一次。

  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值