一、什么是单例模式
保证一个类只有一个实例,并且提供一个访问该全局访问方法。
二、单例模式应用场景
Windows的Task Manager(任务管理器)就是很典型的单例模式,可以试试看你能打开两个windows task manager吗?
Windows的Recycle Bin(回收站)也是典型的单例应用。在整个系统运行过程中,回收站一直维护着仅有的一个实例。
数据库连接池的设计一般也是采用单例模式,因为数据库连接是一种数据库资源。数据库软件系统中使用数据库连接池,主要是节省打开或者关闭数据库连接所引起的效率损耗,这种效率上的损耗还是非常昂贵的,因为何用单例模式来维护,就可以大大降低这种损耗。
三、单利模式类图
四、单利模式的几种创建方式
1、饿汉式:类初始化时,会立即加载该对象,线程天生安全,调用效率高。
public class SingletonDemo1 {
private static SingletonDemo1 singleton = new SingletonDemo1();
private SingletonDemo1() {}
public static SingletonDemo1 getInstance() {
return singleton;
}
}
2、懒汉式:多线程情况下,会产生线程安全问题
public class SingletonDemo2 {
private static SingletonDemo2 singleton;
private SingletonDemo2(){}
public static SingletonDemo2 getInstance() {
if (singleton == null) {
singleton = new SingletonDemo2();
}
return singleton;
}
}
3、双重检验锁
public class SingletonDemo3 {
private static SingletonDemo3 singleton;
private SingletonDemo3(){}
public static SingletonDemo3 getInstance() {
if (singleton == null) {
synchronized (SingletonDemo3.class) {
if (singleton == null) {
singleton = new SingletonDemo3();
}
}
}
return singleton;
}
}
4、静态内部类:多线程环境下推荐使用
public class SingletonDemo4 {
private SingletonDemo4(){}
private static class SingletonInnerClass{
private static SingletonDemo4 singleton = new SingletonDemo4();
}
public static SingletonDemo4 getInstance() {
return SingletonInnerClass.singleton;
}
}
5、使用枚举实现:可以防止反射攻击
public enum SingletonDemo5 {
INSTANCE;
public void mthod() {
System.out.println("hello enum singleton!");
}
}
五、单例防止反射漏洞攻击
可以在构造方法中添加实现一次性判断,但是对方还是可以通过反射修改flag来破坏。
public class SingletonDemo1 {
private static boolean flag = false;
public static SingletonDemo1 singleton = new SingletonDemo1();
private SingletonDemo1() {
if (!flag){
flag = true;
}else {
throw new RuntimeException("单例模式被入侵!");
}
}
public static SingletonDemo1 getInstance() {
return singleton;
}
}
具体使用那种可以根据项目需要进行选择。