上篇文章介绍了单例模式,多例模式,有不明白的同学可以点进去先观看:
面试官:spring单例模式,多例模式,懒汉模式,饿汉模式(一)?
这篇文章重点介绍饿汉模式懒汉模式,
饿汉模式:在加载对象时候,对象就会创建实例,为所有spring配置文件中定义的bean都是生成的一个实例,天生线程安全的,多线程的情况下也不会出现问题。
懒汉模式:在获取对象第一次请求的时候,才会创建实例。本身是线程不安全的,但有几种实现线程安全的写法。
1、饿汉模式:
因为实例被static和final修饰,在对象加载到内存的时候初始化,所以线程安全。
public class HungrySingleton {
private String name;
private static final HungrySingleton hungrySingleton = new HungrySingleton("张三");
public HungrySingleton(String name){
this.name = name;
}
public static HungrySingleton getInstance(){
return hungrySingleton;
}
}
虽然这样写线程安全,但他还是有缺陷的,在getBean实例之前,不能不给他设置属性和参数,这时候懒汉模式就出现了,可以通过双重检索可以实现线程安全。
2、懒汉模式:
private String name;
private static LazySingleton lazySingleton;
public LazySingleton(String name) {
this.name = name;
}
public static LazySingleton getInstance() {
//第一次访问的时候没有对象,所以获取对象
if (lazySingleton == null) {
lazySingleton = new LazySingleton("张三");
}
return lazySingleton;
}
public static LazySingleton getInstance2() {
//保证线程安全
synchronized (lazySingleton) {
if (lazySingleton == null) {
lazySingleton = new LazySingleton("张三");
}
}
return lazySingleton;
}
public static LazySingleton getInstance3() {
//保证线程安全,性能提升
if (lazySingleton == null) {
synchronized (lazySingleton) {
if (lazySingleton == null) {
lazySingleton = new LazySingleton("张三");
}
}
}
return lazySingleton;
}
懒汉模式可以通过双重效验和synchronized来实现线程安全
问:为什么要双重效验?从代码里我们可以看到
getInstance()方法可以实现,第一次访问的时候没有对象,所以为null的时候,获取实例对象,但这种情况下多线程访问时候,会出现异常,导致创建多个实例,如何解决呢?
getInstance2()方法可以保证线程安全,上锁之后,其他线程不可以进入,但这种情况会出现什么问题呢?会一直上锁,导致没必要的性能开销,实际只需要在第一次创建的上锁。
getInstance3()这就是为什么要用双重效验,先判断是否为null,然后在用synchronized上锁实现线程安全。