Java单例模式实现方式
懒汉式-非线程安全
public class LazyNoSafe {
private static LazyNoSafe instance;
public static LazyNoSafe getInstance() {
if (instance == null) {
instance = new LazyNoSafe();
}
return instance;
}
private LazyNoSafe() {}
}
说明:
- 延迟加载
- 非线程安全
- 无法保证序列化和反序列化单例
懒汉式-线程安全
public class LazySafe {
private static LazySafe instance;
public synchronized static LazySafe getInstance() {
if (instance == null) {
instance = new LazySafe();
}
return instance;
}
private LazySafe() {}
}
说明:
- 延迟加载
- 线程安全
- 无法保证序列化和反序列化单例
- 效率低下,synchronized很影响性能
懒汉式-静态内部类
public class LazyUseInner {
private static class InnerClass {
private static LazyUseInner instance = new LazyUseInner();
}
public static LazyUseInner getInstance() {
return InnerClass.instance;
}
private LazyUseInner() {}
}
说明
- 延迟加载
- 线程安全
- 无法保证序列化和反序列化单例
懒汉式-双重锁机制
public class DoubleLock {
private volatile static DoubleLock instance;
public static DoubleLock getInstance() {
if (instance == null) { //第一次检查
synchronized(DoubleLock.class) {
if (instance == null) {
instance = new DoubleLock();
}
}
}
return instance;
}
private DoubleLock() {}
}
说明:
- jdk1.5及以上线程安全,volatile语义在jdk1.5得到修正
- 延迟加载
- 效率不错,由于有第一步检查,规避了绝大多数执行锁同步情况
- 无法保证序列化和反序列化单例
饿汉式
public class HungrySafe {
private static HungrySafe instance = new HungrySafe();
public static HungrySafe getInstance() {
return instance;
}
private HungrySafe() {}
}
说明:
- 简单粗暴
- 无法延迟加载
- 线程安全
- 无法保证序列化和反序列化单例
枚举单例
public enum UseEnum {
INSTANCE;
public void otherMethod() {
// do something...
}
}
说明
- 无法延迟加载
- 线程安全
- 保证序列化和反序列化单例
- Effective Java中推荐使用
结尾
对于序列化的反序列化可以实现readResolve()来保证单例。
项目中使用单例前最好先思考以下三点:线程安全、延迟加载、序列化与反序列化安全。