我是靠谱客的博主 忧虑星月,最近开发中收集的这篇文章主要介绍spring源码系列(五)——番外篇回答网友的问题,觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

回答一下@椰大大�的问题;他在第四篇的文章评论里面留言了;但是他的问题比较复杂;为了把问题讲清楚就挪到这里来回答吧;

在这里插入图片描述

问题原文:

有个问题在网上找半天,问了一堆人也不会,只能留言请教你了。。。 为何@Autowired可以注入ApplicationContext, 一般来说,我们可以通过实现ApplicationContextAware接口来获取ApplicationContext的引用。但是根据官方文档,发现也可以通过 @Autowired来注入ApplicationContext,这是为什么呢?

问题1——去了哪里?

如果你提问我肯定回答;所以肯定还在CSDN。有空肯定奋笔疾书写完spring源码分析的博客

为何@Autowired可以注人 ApplicationContext对象?

这个问题比较复杂;得到ApplicationContext对象的方法多了去了,不止这两种,我就不一一例举了;比如extends WebApplicationObjectSupport也可以获取ApplicationContext对象;
而且可能你有两重意思
1、有了ApplicationContextAware已经得到ApplicationContext为什么还需要@Autowried?
2、 为何@Autowired可以注人 ApplicationContext对象,原理是什么?
我都回答一下吧——正文开始

获取AppliationContext的主流方式

为了适合更对spring还不那么熟悉的读者我先把主流的两种获取ApplicationContext对象的方式列出来;
第一种,在一个bean(注意一定得是bean,被spring容器管理的对象)当中通过@Autowired来注入ApplicationContext对象;代码如下:

在这里插入图片描述

@Component
public class X {
	@Autowired
	Y y;

	@Autowired
	ApplicationContext applicationContext;

	public X(){
		System.out.println("x 的构造方法");
	}
}

第二种,通过实现ApplicationContextAware接口来获取AppliationContext对象;

@Comonpent
public class Util implements ApplicationContextAware{
	private static ApplicationContext applicationContext = null;
	public  void setApplicationContext(ApplicationContext applicationContext){
		Util.applicationContext =applicationContext;
	}
	
	//可以对getBean封装
	public static Object getBean(String name){
		sout("可能你需要干一点其他事吧");
    	return getApplicationContext().getBean(name);
  }
	//也可以提供一个静态方法返回applicationContext
	代码省略了
}

两种获取ApplicationContext对象的方式有什么区别呢?

@Autowried和ApplicationContextAware都能实现得到一个ApplicationContext(再次说明,其实不止这两种方式;但是其他的方式可以忽略,全部归类为接口方式);那么这两种方式有什么区别呢?我认为没什么区别,无非是一个耦合是@Autowried这个注解;另一个耦合的是一个接口;现在的spring版本已经5.x了;如果你们公司使用的spring不支持注解那么就使用接口吧;

为什么需要注入这个ApplicationContext对象呢?

我列举一下官网上说的经典场景吧;

假设类A(单例的)需要注入一个类B(原型的);如果我们直接写会有问题;

比如你在A类当中的m()方法中返回b,那么无论你调用多少次a.m();返回的都是同一个b对象;就违背b的原型规则,应该在m方法中每次都返回一个新的b;所以某些场景下b不能直接注入;

错误的代码:

@Component
public class A{
	
	//注意B是原型的  scope=prototype
	@Autowried;
	B b;
	public B m(){

		//直接返回注入进来的b;肯定有问题
		//返回的永远是A实例化的时候注入的那个bean
		//违背的B设计成原型的初衷
		return b;
	}
}

正确的写法

@Component
public class A{
	@Autowired
	ApplicationContext applicationContext;
	public B m(){
		//每次调用m都是通过spring容器去获取b
		//如果b是原型,每次拿到的都是原型b
		B b= applicationContext.getBean("b");
		return b;
	}
}

当然这个不是我胡说,这是官网的上面的经典例子
官网参考:https://docs.spring.io/spring/docs/5.2.4.RELEASE/spring-framework-reference/core.html#beans-factory-method-injection

In most application scenarios, most beans in the container are
singletons. When a singleton bean needs to collaborate with another
singleton bean or a non-singleton bean needs to collaborate with
another non-singleton bean, you typically handle the dependency by
defining one bean as a property of the other. A problem arises when
the bean lifecycles are different. Suppose singleton bean A needs to
use non-singleton (prototype) bean B, perhaps on each method
invocation on A. The container creates the singleton bean A only once,
and thus only gets one opportunity to set the properties. The
container cannot provide bean A with a new instance of bean B every
time one is needed.

在这里插入图片描述

读者可以自己打开这一章节;可以自行翻译一下;上面是笔者蹩脚的英文水平的理解;说不定你有更加深刻的理解也说不定;

再说一个大家常见的场景

比如A这个类不是被spring管理的对象;但是需要得到一个spring管理的B;这个时候当然不能用传统的@Autowired来注入B,因为A都不是bean,你在A里面写spring注解都不可能生效;只能获取ApplicationContext对象,然后调用getBean方法得到b;

当然可能还有更多场景需要用到这个对象就不再啰嗦了;

为什么@Autowired能够注入ApplicationContext对象?

比如上文中的X里面注入了一个Y;注入了一个ApplicationContext对象;为什么这个ApplicatonContext可以注入成功?这个问题其实很经典;可能有人会想当然的认为肯定可以注入啊;就像注入Y一样简单;假设你对spring有一点点了解你便知道Y之所以能够被注入是因为Y本身就存在spring容器当中;换句话说Y在单例池当中;那么ApplicationContext对象究竟在不在spring容器当中?或者在不在单例池当中呢?

要搞清这个问题的话只有一个办法 Debug It;

首先我们假设ApplicationContext 这个对象也存在spring容器当中;其实如果你读过前面的博客就知道一个bean如果存在spring容器当中,大部分(有的是直接把对象put到单例池,故而没有BeanDefinition)的情况下会有一个与之对应的BeanDefinition对象;也存在容器当中;那么我们看看这个ApplicationContext的BeanDefinition对象有没有呢?

AnnotationConfigApplicationContext ac
			= new AnnotationConfigApplicationContext(App.class);
			
	String[] beanDefinitionNames = ac.getBeanDefinitionNames();
	//打印spring容器当中所有bean的bd
	for (String beanDefinitionName : beanDefinitionNames) {
		System.out.println(beanDefinitionName);
	}
}

执行结果

在这里插入图片描述
运行结果是没有与之对应的beanDefinition对象存在容器;说穿了ApplicationContext这个对象不在容器当中;需要注意的是我这里说的容器这个概念特别庞大,这里不展开讲;其实按照我的理解ApplicationContext这个对象是在容器当中的;但是由于之前文章没有去系统的聊过什么叫做容器,那么大家就先按照自己的理解去看待容器;如果按照一般的理解,看到这里可以认为ApplicationContext不在容器当中吧(再次说明我理解的是在容器当中的);以后再来解释容器这个概念;

如果上面的结果不足以说服你,那么笔者再列出一个证据;
试想一下我们注入的这个ApplicationContext对象肯定单例的;如果每次注入的ApplicationContext都是一个新的那肯定不合理;ApplicationContext如果是单例的讲道理他会存在单例池当中;所以我们可以看看单例池是否存在这个对象;

在这里插入图片描述
可以看到连Y都被实例化并且存在单例池里面了(这也是Y之所以能够用@Autowried注入进来的原因),但是没有看到ApplicationContext对象;那么可以肯定ApplicationContext对象不在spring容器当中(再再再再次说明,其实笔者认为他是存在容器当中的;可能我对容器的理解可能更深一点吧);既然他不在容器当中,也就是和Y不一样;怎么注入进来的呢?

如果你系统的阅读过spring源码就会知道完成@Autowried这个注解功能的类是AutowiredAnnotationBeanPostProcessor这个后置处理器;说穿了对@Autowried这个注解的解析就是这个后置处理器;我们可以看看他是如何完成@Autowried解析、注入属性的;找到这个后置处理器当中完成属性注入的方法——debug it;

postProcessProperties方法便是处理属性注入的方法;

在这里插入图片描述
图上可以看到X已经实例化成对象了(还不是bean),但是里面的属性都是null,因为他才刚刚开始来填充属性;调用metadata.inject(bean, beanName, pvs);继续完成填充属性——debug it

metadata.inject(bean, beanName, pvs);首先会遍历当前bean——当中所有需要注入的属性;也就是有两个Y 和 ApplicationContext;
所以第一次for循环注入的是y;就是从单例池当中获取一下Y;如果获取不到就实例化Y放到单例池然后返回;反射填充属性Y;完成注入;
在这里插入图片描述
当然我们关注的是ApplicationContext的注入;所以调试第二次循环——element==ApplicationContext的时候

在这里插入图片描述

完成ApplicationContext的步骤—— 多图警告:
1、调用element.inject(target, beanName, pvs);
在这里插入图片描述
2、调用beanFactory.resolveDependency方法得到ApplicationContext对象;主要就是这里怎么得到的?他不在单例池当中为何可以获取到?
在这里插入图片描述
3、调用doResolveDependency方法获取ApplicationContext对象;

在这里插入图片描述

4、调用findAutowireCandidates获取ApplicationContext对象
在这里插入图片描述

5、关键代码了

在这里插入图片描述
贴一下主要代码吧

Map<String, Object> result = new LinkedHashMap<>(candidateNames.length);
	for (Map.Entry<Class<?>, Object> classObjectEntry : this.resolvableDependencies.entrySet()) {
		Class<?> autowiringType = classObjectEntry.getKey();
		if (autowiringType.isAssignableFrom(requiredType)) {
			Object autowiringValue = classObjectEntry.getValue();
			autowiringValue = AutowireUtils.resolveAutowiringValue(autowiringValue, requiredType);
			if (requiredType.isInstance(autowiringValue)) {
				result.put(ObjectUtils.identityToString(autowiringValue), autowiringValue);
				break;
			}
		}
	}

对这段代码做一点解释吧

当代码执行到findAutowireCandidates的时候,传了一个特别重要的参数 Class requireType;就是当前注入的属性的类型——也就是ApplicationContext.class;

然后遍历了一个map——resolvableDependencies(关于这个map,下文有解释)

for (Map.Entry<Class<?>, Object> classObjectEntry : this.resolvableDependencies.entrySet()) {

接着会把传过来的requireType和遍历出来的autowiringType——当前类型进行比较

if (autowiringType.isAssignableFrom(requiredType)) {

如果传过来的requiredType和遍历出来Class对象相同则停止遍历,直接把当前遍历出来的对象返回作为注入属性的值,完成属性注入;

如果你调试代码可以看到这个map——resolvableDependencies一共就四个对象;

在这里插入图片描述
这个四个对象就包含了一个ApplicationContext,所以在@Autowired 注入ApplicationContext的时候这个for循环会进入,并且直接返回了map当中已经存在好的ApplicationContext对象以便完成属性的注入;但是如果普通bean的注入,比如X注入Y,这不会进入这个for循环;我们可以证明一下;

注入ApplicationContext的时候:
在这里插入图片描述
可以看到for循环已经进入,并且判断成功 autowiringValue已经有值了,进入if,break然后返回这个autowiringValue,完成属性注入;

再来看看普通属性y的注入和这个是否有差别呢?

在这里插入图片描述

这个时候就是x注入y,继续debug会发现整个for循环当中的if是不会进的;也就是相当于这个for循环没有任何作用;

在这里插入图片描述
可以看到findAutowireCandidates已经执行完了,但是需要注入的属性y还只是一个class,不是对象;后台也只打印了x的构造方法没有打印y的;说穿了到到此为止y还是没有找到;

但是如果需要注入的属性是ApplicationContext这里得到的就不一样,因为上面已经说了for循环里面已经返回了对象;
证明一下吧
在这里插入图片描述
那么现在Y(对象或者bean)是如何获得并返回的?

其实这个不是这里讨论的,我如果能够把spring源码系列文章写完会会写到属性注入的全部过程和原理的;

这里先给个基本结果吧

在这里插入图片描述

也就是上图这个代码这里把Y的对象或者叫做bean获取到返回出去完成属性注入;

说了这么多总结一下吧:

普通对象的注入如果注入的属性是单例,那么spring首先从单例池获取,如果获取不到直接实例化这个bean,放到单例池在返回,完成注入;如果是原型的每次注入就直接实例化这个原型bean返回完成注入;

ApplicationContext对象的注入不同,如果注入的属性是ApplicationContext类型,那么spring会先从resolvableDependencies这个map当中去找,如果找到直接返回一个ApplicationContext对象完成属性注入;

那么问题来了,resolvableDependencies这个map的作用是什么
他其实有一个javadoc
在这里插入图片描述

/** Map from dependency type to corresponding autowired value. */
	//缓存一些依赖类型通用自动注入的值
	private final Map<Class<?>, Object> resolvableDependencies = new ConcurrentHashMap<>(16);

以我蹩脚的英文水平理解这个map缓存一些依赖类型通用自动注入的值;也就是@Autowried的时候有一些通用或者常用的类型的值都存放这个map里面;

如果这个javadoc不是很详细可以参考另外一个;这个map其实spring没有开发给程序员使用,private的;那么可想而知他肯定提供了api来操作这个map;找到他,看看这个api的说明

在这里插入图片描述

Register a special dependency type with corresponding autowired value
注册一个特殊的依赖类型——通用的注入的值
换句话说比如你有一些通用的对象、可能会被别的bean注入,那么你可以调用这个方法把这写对象放到一个map当中——resolvableDependencies
下面还有更加详细的说明,意思说spring当中的一些工厂或者上下文对象他们在bean工厂里面不是定义为bean,这个时候如果别的bean需要注入,则可以把他们放到这个map当中;

那么spring什么时候把这四个对象放到这个map当中的呢?——spring容器初始化的时候
我先把调用链列出来,然后在截几个图说明一下:
0、main方法
1、org.springframework.context.support.AbstractApplicationContext#refresh
2、org.springframework.context.support.AbstractApplicationContext#prepareBeanFactory
3、org.springframework.beans.factory.config.ConfigurableListableBeanFactory#registerResolvableDependency

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

这就是为什么我们可以注入ApplicationContext的全部原因吧;

最后

以上就是忧虑星月为你收集整理的spring源码系列(五)——番外篇回答网友的问题的全部内容,希望文章能够帮你解决spring源码系列(五)——番外篇回答网友的问题所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(40)

评论列表共有 0 条评论

立即
投稿
返回
顶部