单例模式
优点
1、在内存里只有一个实例,减少了内存的开销,尤其是频繁的创建和销毁实例。
2、避免对资源的多重占用(比如写文件操作)。
缺点
没有接口,不能继承,与单一职责原则冲突,一个类应该只关心内部逻辑,而不关心外面怎么样来实例化。
使用场景
1、要求生产唯一序列号。
2、WEB 中的计数器,不用每次刷新都在数据库里加一次,用单例先缓存起来。
3、创建的一个对象需要消耗的资源过多,比如 I/O 与数据库的连接等。
实现方式
1 懒汉式
注
这种方式采用双锁机制,安全且在多线程情况下能保持高性能。
getInstance() 的性能对应用程序很关键。
public class LazySingleton {
//没有volatile其他线程可能由于指令重排访问到还没有初始化的对象
//volatile变量规则:对一个变量的写操作先行发生于后面对这个变量的读操作;
private static volatile LazySingleton lazySingleton = null;
private LazySingleton(){
}
public static LazySingleton getInstance(){
//双检锁/双重校验锁(DCL,即 double-checked locking)
if (lazySingleton == null){
synchronized (LazySingleton.class){
if (lazySingleton == null){
lazySingleton = new LazySingleton();
}
}
}
return lazySingleton;
}
}
2 饿汉式
注
优点:没有加锁,执行效率会提高。
缺点:类加载时就初始化,浪费内存。
public class Singleton {
private static Singleton singleton = new Singleton();
private Singleton(){}
public static Singleton getInstance(){
return singleton;
}
}