单例模式:简单来说,就是一个类只能构建一个对象的设计模式
单例模式(非线程安全):
public class LanHan {
private static LanHan lan = null;
private LanHan(){}
public static LanHan getInstance(){
if(lan == null){
lan = new LanHan();
}
return lan;
}
}
解释:
1.要想让一个类只能构建一个对象,自然不能让它随意去做new操作,因此LanHan的构造方法只能是私有的。
2.getInstance()是获取单例对象的方法
(上述模式属于单例模式中的懒汉模式,即 需要时再实例化,对应的饿汉模式为,在判空操作前,之前new对象 PS:这种模式的实例化对象不论是否需要,一直存在,占用资源)
3.上述代码存在的问题,线程不安全。假设该类刚刚创建,这时LanHan对象还是空,有两个线程同时访问getInstance()方法,判空之后,都会执行new操作。所以为非线程安全的,那我们自然就会想到加锁,下面就让我们对代码做一些修改
单例模式(双重检查机制):
public class LanHan3 {
private static LanHan3 lan;
private LanHan3(){
}
public static LanHan3 newInstance(){
if(lan==null){
synchronized(LanHan3.class){//同步锁
if(lan==null){
lan = new LanHan3();
}
}
}
return lan;
}
}
解释:
1.加锁是为了防止new多个对象。
2.进行二次判断的原因:当两个线程同时进入的话,第一个线程已经实例化对象了,如果不进行判断的话,第二个线程也将实例化一个对象。
小结:
关于懒汉模式与饿汉模式的应用场景:在不考虑反射构造的情况下,懒汉模式是通过时间换空间,饿汉模式是通过空间换时间。懒汉模式可以通过Synchronized关键字保证线程安全,饿汉模式通过类加载机制保证线程安全。