单例设计模式在Spring中的应用
一.实现一个单例
可以使用如下的步骤实现一个单例类: 单例设计模式的实现流程 1、将构造方法私有化,使用private关键字修饰。使其不能在类的外部通过new关键字实例化该类对象。 2、在该类内部产生一个唯一的实例化对象,并且将其封装为private static类型。 3、对外提供一个静态方法getInstance()负责将对象返回出去,使用public static修饰
首先单例方法的构造方法是一个私有的方法,不可以通过构造器的方式进行构建,只能通过getInstance()方法获取单例。 下边的if()判断,判断是不是构造过,如果构造了就直接返回。这种需要在使用的时候,调用才会构建这种方式就是懒加载,需要了使用时构建,不会浪费资源。但是不是线程安全的。 加了以后线程是安全的了,但是锁的粒度太大 把syc加到方法里 如果第一个线程发现成员变量为null,准备创建对象;这是第二 个线程同时也发现成员变量为null,也会创建新对象。这就会造成在一个JVM中有多个单例类型的实例 再加入一个判空操作 由于可见性和指令重排,还需要加上volatile标识
二,设计模式及其在Spring中的应用
单例模式定义: 保证一个类仅有一个实例,并提供一个访问它的全局访问点。 双重检查锁实现单例模式Code
// double-check 双重检查锁 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){ //参数true设置标识允许早期依赖 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) { //如果此bean正在加载,则不处理 singletonObject = this.earlySingletonObjects.get(beanName); if (singletonObject == null && allowEarlyReference) { //当某些方法需要提前初始化的时候则会调用addSingleFactory 方法将对应的ObjectFactory初始化策略存储在singletonFactories ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName); if (singletonFactory != null) { //调用预先设定的getObject方法 singletonObject = singletonFactory.getObject(); //记录在缓存中,earlysingletonObjects和singletonFactories互斥 this.earlySingletonObjects.put(beanName, singletonObject); this.singletonFactories.remove(beanName); } } } } return (singletonObject != NULL_OBJECT ? singletonObject : null); }