1.单例模式介绍
整个程序中只创建一个实例,构造方法私有化,只能自己创建自己的唯一实例,提供一个全局的获取唯一实例方法。节省内存开销,避免频繁地创建与销毁对象。
2.单例模式实现方式
2.1懒汉式
线程不安全,延迟加载
public class Singleton {
private static Singleton instance;
private Singleton() {
}
public static Singleton getInstance() {
if (instance == null) {
instance = new Singleton();
}
return instance;
}
}
2.2懒汉式
线程安全,延迟加载
public class Singleton {
private static Singleton instance;
private Singleton() {
}
public static synchronized Singleton getInstance() {
if (instance == null) {
instance = new Singleton();
}
return instance;
}
}
2.3饿汉式
线程安全,立即加载
,线程安全由jvm保证,因为类只会加载一次,类加载后会即刻加载该静态属性。
类加载的时候就会初始化,不介意内存推荐使用此种方式。
public class Singleton {
private static Singleton instance = new Singleton();
private Singleton() {
}
public static Singleton getInstance() {
return instance;
}
}
2.4双重校验锁
线程安全,延迟加载
,写起来比较复杂,还涉及到volatile
public class Singleton {
private volatile static Singleton singleton;
private Singleton() {
}
public static Singleton getSingleton() {
if (singleton == null) {
synchronized (Singleton.class) {
if (singleton == null) {
singleton = new Singleton();
}
}
}
return singleton;
}
}
2.5静态内部类
线程安全,延迟加载
,算是比较完美的写法了,外部类加载时内部类不会加载
,保证了lazy loading,用到的时候内部类才会初始化,且只加载一次,jvm保证线程安全
public class Singleton {
private static class SingletonHolder {
private static final Singleton INSTANCE = new Singleton();
}
private Singleton() {
}
public static final Singleton getInstance() {
return SingletonHolder.INSTANCE;
}
}
2.6枚举
枚举设计即保证单例,且线程安全
,还可以防止反序列化
,可以说是最完美的单例了。
public class Singleton {
private Singleton() {
}
static enum SingletonEnum {
INSTANCE;
private Singleton Singleton;
private SingletonEnum() {
Singleton = new Singleton();
}
public Singleton getInstance() {
return Singleton;
}
}
public static Singleton getInstance() {
return SingletonEnum.INSTANCE.getInstance();
}
}
3.单例模式总结
推荐使用饿汉式、双重校验锁、静态内部类这三种方式。
不考虑内存消耗建议使用饿汉式,考虑使用延迟加载的话,可使用双重校验锁和静态内部类。
实际项目运用时也不必纠结具体使用哪种,哪种方便、简单、合适就用哪种,如果想不起来用哪一种那就用 饿汉式
即简单又线程安全。