什么是单例模式?
- 单例类只能有一个实例
- 单例类必须自己创建自己的唯一实例
- 单例类必须给所有其他对象提供这一实例
使用单例模式的目的?
单例模式主要作用是保证在Java应用程序中,一个类Class只有一个实例存在
懒汉式
线程不安全单例模式
public class Singleton {
private static Singleton instance;
private Singleton (){}
public static Singleton getInstance() {
if (instance == null) {
instance = new Singleton();
}
return instance;
}
}
缺陷: 线程不安全,如果两个线程同时访问到了getInstance()方法,并且都走到了“instance == null”这个判断时,此时instance实例未被创建,所以两个线程都会创建一个实例
改进: 加上同步锁
public class Singleton {
private static Singleton instance;
private Singleton (){}
public static synchronized Singleton getInstance() {
if (instance == null) {
instance = new Singleton();
}
return instance;
}
}
优点: 第一次调用才初始化,避免内存浪费
缺点: 必须加锁 synchronized 才能保证单例,但加锁会影响效率
恶汉式
public class Singleton {
private static Singleton instance = new Singleton();
private Singleton (){}
public static Singleton getInstance() {
return instance;
}
}
饿汉式更为安全,在类加载的时候就已经自行创建了实例,getinstance方法里面一直引用的就是这个实例,而且永远不会释放,一直存在内存中,知道程序结束
优点: 没有加锁,执行效率会提高。
缺点: 类加载时就初始化,浪费内存。
懒汉式 双重锁
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;
}
}
- 其中用了两个if 判断,第一个if 先判断singleton是否为null,如果不为null,说明singleton 已经被初始化了,直接返回singleton
- 如果singleton 为null ,说明singleton 还没有被初始化,这样才会去执行synchronized 修饰的代码块内容,只在其初始化的时候调用一次。
- 这样的设计既能保证只产生一个实例,并且只在初始化的时候加同步锁,也实现了延迟加载。