采用懒加载的方式获取实例
class Singleton3{
private static Singleton3 instance;
private Singleton3(){
}
//提供一个静态公有的方法,当使用的时候才实例化,实现懒汉式加载
public static Singleton3 getInstance(){
if(instance == null){
instance = new Singleton3();
}
return instance;
}
}
当需要实例化对象的时候,先判断该对象是否存在,如果存在直接返回,没有的话才创建实例,这种方式可以在使用的时候才实例,不会造成内存的浪费。
但是这种方式只能是单线程才能使用,因为在多线程的环境下,在判断对象是否为空的时候可能会被其他线程打断而造成的对象的不一致。因此在实际开发中不使用这种方式。
改进(加锁)
既然会造成不安全,那么我们就在这个基础上添加一个锁,避免线程的中断
class Singleton4{
private static Singleton4 instance;
private Singleton4(){
}
//提供一个静态公有的方法,加入同步处理代码,解决线程安全问题
// 当使用的时候才实例化,实现懒汉式加载
public static synchronized Singleton4 getInstance(){
if(instance == null){
instance = new Singleton4();
}
return instance;
}
}
这种方式可以解决线程不安全的问题,只是这样会造成效率不高,每次都要进行同步操作,造成不必要的资源浪费。
再改进(双重检查)
因为获取实例的时候只需要实例化一次,后面获取只需要返回实例即可,每次的加锁会造成资源的浪费,因此可以在代码块中进行判断,如果没有实例化,那就进行实例化,但是在此这前还需要进行再一次的判断
class Singleton5{
private static volatile Singleton5 instance;
private Singleton5(){
}
//提供一个静态公有的方法,加入双重检查代码,解决线程安全问题。同时实现懒加载问题
// 当使用的时候才实例化,实现懒汉式加载
public static Singleton5 getInstance(){
if(instance == null){
synchronized (Singleton5.class){
if(instance == null){//多线程的判断
instance = new Singleton5();
}
}
}
return instance;
}
}
这种方法推荐使用。
优点:线程安全,懒加载,效率高,避免了反复加锁造成的资源浪费。