Java单例的正确使用姿势

Java单例模式可能是最简单也是最常用的设计模式,一个完美的单例需要做到哪些事呢?

  1. 单例(这不是废话吗)
  2. 延迟加载
  3. 线程安全
  4. 没有性能问题
  5. 防止序列化产生新对象
  6. 防止反射攻击

可以看到,真正要实现一个完美的单例是很复杂的,那么,让我这个司机带大家看一看正确姿势的单例。

最佳实践单例之枚举
没错,直接就上最佳实践,就是这么任性

这货长这样:

public enum Singleton{
    INSTANCE;
}

如果你不熟悉枚举,可能会说:这货是啥?!

这种方式的好处是:

  1. 利用的枚举的特性实现单例
  2. 由JVM保证线程安全
  3. 序列化和反射攻击已经被枚举解决

    调用方式为Singleton.INSTANCE, 出自《Effective Java》第二版第三条: 用私有构造器或枚举类型强化Singleton属性。

关于单例最佳实践的讨论可以看Stackoverflow:what-is-an-efficient-way-to-implement-a-singleton-pattern-in-java

下面将会介绍更为常见的单例模式,但是均未处理反射攻击,如果想了解更多可以看这篇文章:如何防止单例模式被JAVA反射攻击

public class Singleton {
    private static final Singleton INSTANCE = new Singleton();
    // 私有化构造函数
    private Singleton(){}

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

这种单例的写法最简单,但是缺点是一旦类被加载,单例就会初始化,没有实现懒加载。而且当实现了Serializable接口后,反序列化时单例会被破坏。

实现Serializable接口需要重写readResolve,才能保证其反序列化依旧是单例:

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值