懒汉式单例模式双重检测锁的理解

public class Singleton {
    //volatile 防止指令重排 和可见性
    private volatile static Singleton uniqueInstance;

    private Singleton() {
    }

    public Singleton getUniqueInstance() {
        //先判断对象是否已经实例化过,没有实例化才进入加锁代码
        if (uniqueInstance == null) {
            //类对象加锁
            synchronized (Singleton.class) {
                //避免 singleTon== null时,第一个线程实例化后,进入阻塞状态的线程被唤醒后仍会进行实例化。
                if (uniqueInstance == null){
                    uniqueInstance = new Singleton();
                }
            }
        }
        return uniqueInstance;
    }
}

  1. 第一个 if (uniqueInstance == null) 是为了判断对象是否已经实例化,只有非实例化的情况下才能进入if块进行加锁。
  2. 第二个if(uniqueInstance == null) 是因为:假设存在线程1和线程2,当第一次实例化时,他们都能进入到第一个if块中,当线程1获取到Singleton的锁,由于此时uniqueInstance 还未加载可以进入第二个if块,当执行完uniqueInstance = new Singleton(); 便返回 uniqueInstance,此时uniqueInstance已经实例化,但是此时线程2获得锁,如果没有第二个if块则会再次new 一个新的实例出来,就违反了单例的规则。
  • 3
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
以下是懒汉单例模式和饿汉单例模式的异同: 相同点: 1. 都是单例模式,即保证一个类只有一个实例对象。 2. 都使用了私有的构造函数,防止外部创建实例对象。 3. 都使用了静态变量来保存实例对象。 不同点: 1. 创建对象的时机不同:饿汉在类加载时就创建了对象实例,而懒汉是在使用时才创建。 2. 线程安全性不同:饿汉天生是线程安全的,因为在类加载时就已经创建了对象实例,而懒汉需要考虑线程安全问题,可以使用synchronized关键字或者双重检查定等方来保证线程安全。 3. 性能不同:饿汉在类加载时就创建了对象实例,所以在访问速度和反应时间上都比懒汉快,但是如果这个实例一直没有被使用,那么就会造成内存浪费。而懒汉只有在使用时才会创建对象实例,所以在内存占用上比饿汉要低,但是在访问速度和反应时间上会稍微慢一些。 下面是懒汉单例模式的示例代码: ```python class Singleton: __instance = None def __init__(self): if Singleton.__instance != None: raise Exception("该类已经实例化过了") else: Singleton.__instance = self @staticmethod def getInstance(): if Singleton.__instance == None: Singleton() return Singleton.__instance ``` 下面是饿汉单例模式的示例代码: ```python class Singleton: __instance = Singleton() def __init__(self): if Singleton.__instance != None: raise Exception("该类已经实例化过了") else: Singleton.__instance = self @staticmethod def getInstance(): return Singleton.__instance ```

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值