java单例模式的各种实现以及DCL的注意事项

饿汉式

class Singleton {
    private static final Singleton1 instance = new Singleton1();

    private Singleton() {

    }

    public static Singleton getInstance() {
        return instance;
    }
}

要点:
1.私有化构造器
2.创建唯一的一个对象,并提供获取方法

缺点:
类加载的时候就会创建实例,但是不一定会用到,有可能会造成内存的浪费

懒汉式:同步方法

class Singleton {
    private static Singleton instance;

    private Singleton() {

    }

    public static synchronized Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
        return instance;
    }
}

懒汉式:DCL

class Singleton {
    private static volatile Singleton instance;  // 加volatile是为了禁止指令重排序,防止得到半成品实例
    private static final Object o = new Object();

    private Singleton() {

    }

    //  DCL双重检查
    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (o) {
                if (instance == null) {
                    instance = new Singleton();
                    return instance;
                }
            }
        }
        return instance;
    }
}

解释一下加volatile的作用:
首先创建一个对象大致有三步:1.分配一块内存空间;2.调用构造方法初始化;3.建立引用变量和对象之间的引用关系。
如果不加volatile,则可能发生指令重排序,导致第二步和第三步的执行顺序发生交换,导致有可能拿到一个半成品对象,如下图:
在这里插入图片描述
两次检查的作用:
如果只有一次检查是否为null,则可能不再是单例,如下图:
在这里插入图片描述

静态内部类

class Singleton {
    private Singleton() {

    }

    public static Singleton getInstance() {
        return SingletonInstance.instance;  // 多个线程拿到的都是同一个
    }

    private static class SingletonInstance {
        //  因为是static所以只会被执行一次,从而保证单例
        //  有点类似于饿汉式的思想(不过是懒加载的)
        private static final Singleton instance = new Singleton();
    }
}

静态内部类的特点:
1.外面的类加载的时候,静态内部类并不会加载
2.静态内部类只会被加载一次,而且加载过程是线程安全的

枚举的方式

enum Singleton7 {
    INSTANCE;
}

枚举里只有一个对象,天然就是单例的。

如果你想了解更多我对编程和人生的思考,请关注公众号:青云学斋

公众号

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值