单例模式笔记

单例模式

单例模式:程序运行时,让一个类只能创建一个对象

饿汉式:

  • 构造方法私有化
  • 创建当前类对象
  • 对外提供公共的访问方法将SingletonHungary对象暴露给外部
//单例模式------饿汉式
public class SingletonHungary{
    
    
    //1.将构造方法私有化
    private SingletonHungary(){
    
    
        
    }
    //2.创建当前类的对象
    private static SingletonHungary s = new SingletonHungary();
    //3.对外提供公共访问方法将SingletonHungary对象暴露给外部
    public static SingletonHungary getInstance(){
    
    
        return s;
    }
}

懒汉式

//单例模式----懒汉式,延迟加载模式
public class SingletonLazy {
    
    

    //1.将构造方法私有化
    private SingletonLazy() {
    
    
        System.out.println(Thread.currentThread().getName() + "ok");
    }

    //2.创建当前类的引用
    private static volatile SingletonLazy s;

    //3.对外提供公共的访问方法将SingletonHungary对象暴露给外部
    public static SingletonLazy getInstance() {
    
    
        if (s == null) {
    
    
            synchronized (SingletonLazy.class) {
    
    
                if (s == null) {
    
    
                    s = new SingletonLazy();//不是一个原子性操作
                   /**
                     * 1.分配内存空间
                     * 2.执行构造方法,初始化对象
                     * 3.把这个对象指向这个空间
                     *
                     * 123
                     * 132 A
                     *     B
                     */
                    
                }
            }
        }
        return s;
    }

    //测试多线程
    public static void main(String[] args) {
    
    
        for (int i = 0; i < 100; i++) {
    
    
            new Thread(() -> {
    
    
                SingletonLazy.getInstance();
            }).start();

        }
    }

}


注意:

懒汉式在创建对象的时候不是一个原子性的操作,由于它不是一个原子性的操作,所以它在new 一个对象的时候有三个步骤:

1.分配内存空间内

2.执行构造方法,初始化对象

3.把这个对象指向这个空间

我们期望的是按照123的顺序执行,但在并发情况下,有可能会执行132,这是在电脑的cup上是可以做到的,

A经常进来的时候没有问题(也就是第一次进来的时候),但B进程进来就有问题了,由于它已经指向了这个空间了,B会认为这个对象不是null,就直接返回了,由于B还没完成构造,所以它的这个空间其实是一个虚的,

所以解决这个问题的方法是:必须在对象的引用上加一个volatile关键字,实现变量修改的可见性

猜你喜欢

转载自blog.csdn.net/weixin_46449115/article/details/108418288