概念
所谓单例,就是从头到尾只有一个对象的意思。不过,为什么只能有一个对象呢?我觉得可能有以下原因:
- 对象本身是线程安全的,不需要多个对象;
- 创建多个对象的开销比较大;
单例主要的使用场景在于组件化,比如:spring的bean管理、工具类。
另外,有时候我们可以从单例扩展到多例,这种情况比较常见的场景就是对象池。比如:数据库连接池、线程池等。
示例1,最简单的单例
public class Single {
private static final Single instance = new Single();
public static Single instance() {
return instance;
}
}
示例2,枚举型单例
这种方式能避免反序列化和反射方式创建多个单例的问题。我这儿最推荐这种方式,虽然以枚举的方式实现比较变态
public enum Single {
single;
public void calc() {}
}
示例3,延迟初始化的单例
这儿为啥需要加锁,以及锁里面为啥需要再判断一次null
呢?这是因为多线程里面可以使用锁来保证内存可见性和顺序性。
而至于为啥需要在锁里面再判断一次空指针,是因为被阻塞线程再阻塞线程执行完后,看见的instance已经是非null的,但是没有条件能退出实例化。
public class Single {
private static Single instance;
public static Single instance() {
if (instance == null) {
synchronized (Single.class) {
if (instance == null) {
instance = new Single();
}
}
}
return instance;
}
}
示例4,另外一种延迟初始化的单例
使用内部类加载机制来实现延迟初始化,弊端就是会额外多维护一个类,增加了维护的成本。
public class Single {
public static Single instance() {
return Holder.single;
}
private static class Holder {
private static final Single single = new Single();
}
}
网友评论