1行代码实现高性能单例模式

单例是指:无论你创建了多少个引用,在堆中仅仅只有一个实例。

例如,

Person p1 = ... ; 
Person p1 = ...; 

在单例模式下,必须保证p1==p2。因为单例要保证p1和p2两个引用指向的是同一个实例,即引用地址是相同的。

 

单例模式的写法也有很多种:线程不安全的懒汉式、线程安全的懒汉式、线程安全的饿汉式、登记式等等。但从实践的角度来看,用的最多的也就只有两种方式:枚举式和双重检查式。

 

其中,枚举式的单例模式只需要一行代码,堪称居家旅行必备神器~如下所示。

public enum Singleton{ INSTANCE; }

OK,写完了,不信来测一下。
public enum Singleton{
    INSTANCE;

    public void method(){
        System.out.println("业务方法...");
    }
}

class Test{
    public static void main(String[] args) {
        Singleton ins1 = Singleton.INSTANCE;
        Singleton ins2 = Singleton.INSTANCE;
        System.out.println(ins1 == ins2);
        ins1.method();
    }
}

运行结果:

true

业务方法...

 

原理:枚举类型 是天然的单例。

 

除了以上枚举式的单例模式以外,另一种推荐的单例写法就是 双重检查式单例,源码如下:

public class SingletonD {
    private volatile static SingletonD instance = null;
    private SingletonD() {
    }
    public static SingletonD getInstance() {
        if (instance == null) {
            synchronized (SingletonD.class) {
                if (instance == null)
                    instance = new SingletonD();
            }
        }
        return instance;
    }
}

对以上源码进行两点说明:

  1. 源码中有两个if判断,第一个if是为了减少if代码块的执行次数(如果instance不为null,就不需要再执行if代码块),从而提升性能;第二个if是在加锁后的判断,目的是为了保证instance只会被new一次。这种写法有两个if,因此称为双重加锁。
  2. instance = new SingletonD()不是一个原子性操作,可能会被JVM重排序,从而造成线程安全问题。因此需要给成员变量instance加上volatile关键字,用于防止重排序的发生。

 

 

--- 完 ---

 

10行代码2道题,全答对的你月薪多少了?
一道JVM面试题,答错率超90%
答疑 | synchronized有指令重排序的功能吗?
Ajax - 使用Ajax传递Json数据
【人工智能】Java Web+百度AI实现人脸识别(一)

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值