首先要问自己,什么是单例,为什么要使用单例,在什么情况下使用 java 单例设计模式,不能因为是技术就去学习,要知道学习是为了解决什么问题!
什么是单例?
一个类在系统中只产生一个对象,所有服务都通过这对象提供。
为什么要用单利?
多个实例会引起程序逻辑错误的时候,或者创建某个对象的成本比较高(hibernate 中 SessionFactory),就需要单例。
如何实现单例模式?下面直接上代码
1、饿汉
public class Singleton {
private static Singleton uniqueInstance;
private Singleton() {}
public static Singleton getUniqueInstance() {if (uniqueInstance == null) {uniqueInstance = new Singleton();
}
return uniqueInstance;
}
}
关键在于私有化构造函数,比较节约资源,需要时创建,但是在多线程中不安全。
2、懒汉
public class Singleton {private static Singleton uniqueInstance = new Singleton();
private Singleton() {}
}
线程安全,比较消耗资源
3、懒汉式 - 线程安全
public class Singleton {
private static Singleton uniqueInstance;
private Singleton() {}
public static synchronized Singleton getUniqueInstance() {if (uniqueInstance == null) {uniqueInstance = new Singleton();
}
return uniqueInstance;
}
}
4、双重检查锁
public class Singleton {
private volatile static Singleton uniqueInstance;
private Singleton() {}
public static Singleton getUniqueInstance() {if (uniqueInstance == null) {synchronized (Singleton.class) {if (uniqueInstance == null) {uniqueInstance = new Singleton();
}
}
}
return uniqueInstance;
}
}
线程安全,必须加上 volatile,可以禁止 JVM 的指令重排,保证在多线程环境下也能正常运行。
5、静态内部类,线程安全
public class Singleton {private Singleton() { }
private static class SingletonHolder {private static final Singleton INSTANCE = new Singleton();
}
public static Singleton getUniqueInstance() {return SingletonHolder.INSTANCE;}
}
6、枚举,可以防止反射攻击
public enum Singleton {
INSTANCE;
private String objName;
public String getObjName() {return objName;}
public void setObjName(String objName) {this.objName = objName;}
}
正文完