一.实现一个单例
可以使用如下的步骤实现一个单例类:
单例设计模式的实现流程
1、将构造方法私有化,使用private关键字修饰。使其不能在类的外部通过new关键字实例化该类对象。
2、在该类内部产生一个唯一的实例化对象,并且将其封装为private static类型。
3、对外提供一个静态方法getInstance()负责将对象返回出去,使用public static修饰
首先单例方法的构造方法是一个私有的方法,不可以通过构造器的方式进行构建,只能通过getInstance()方法获取单例。 下边的if()判断,判断是不是构造过,如果构造了就直接返回。这种需要在使用的时候,调用才会构建这种方式就是懒加载,需要了使用时构建,不会浪费资源。但是不是线程安全的。 加了以后线程是安全的了,但是锁的粒度太大 把syc加到方法里 如果第一个线程发现成员变量为null,准备创建对象;这是第二 个线程同时也发现成员变量为null,也会创建新对象。这就会造成在一个JVM中有多个单例类型的实例 再加入一个判空操作 由于可见性和指令重排,还需要加上volatile标识 多线程下的单例模式 单例模式
二,设计模式及其在Spring中的应用
单例模式定义: 保证一个类仅有一个实例,并提供一个访问它的全局访问点。 双重检查锁实现单例模式Code
class Singleton{
private static volatile Singleton instance;
private Singleton(){}
public static sychronize Singleton getInstace(){
if(instance==null){
synchronized(Singleton.class){
if(instance==null) instance=new Singleton();
}
}
return instance;
}
}
ApplicationContext实现原理
Spring依赖注入Bean实例默认是单例的。
Spring的依赖注入(包括lazy-init方式)都是发生在AbstractBeanFactory的getBean里。getBean的doGetBean方法调用getSingleton进行bean的创建。 分析getSingleton()方法
public Object getSingleton(String beanName){
return getSingleton(beanName,true);
}
protected Object getSingleton(String beanName, boolean allowEarlyReference) {
Object singletonObject = this.singletonObjects.get(beanName);
if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
synchronized (this.singletonObjects) {
singletonObject = this.earlySingletonObjects.get(beanName);
if (singletonObject == null && allowEarlyReference) {
ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
if (singletonFactory != null) {
singletonObject = singletonFactory.getObject();
this.earlySingletonObjects.put(beanName, singletonObject);
this.singletonFactories.remove(beanName);
}
}
}
}
return (singletonObject != NULL_OBJECT ? singletonObject : null);
}
ps:spring依赖注入时,使用了双重检查锁实现的单例模式
|