Spring5源码分析之AnnotationConfigApplicationContext
前言: 主要了解的内容有如下几点:
- @Qualifier与@Primary注解的使用
- Spring中ApplicationContext的作用
- BeanFactory与ApplicationContext区别
- AnnotationConfigApplicationContext启动原理分析
假设 IUser两个实现类UserImpl1 和 UserImpl2
1. @Autowired + @Qualifier("UserImpl1") 可以指定使用哪个实现类。 也可以 @Resource("UserImpl1 ")
2. @Primary是指定加载谁,用在实现类上。指定。
图解:
ApplicationContext:也就是上下文
1. AnnotationConfigApplicationContext
2. ClassPathXmlApplicationContext
ApplicationContext 继承BeanFactory 实现扩展功能
小结:
@Configuration 相当于XML配置
注解方式启动,需要配置config,通过构造函数去加载config的配置信息
看下这个构造函数:
注意先执行父类的无参构造函数 再执行 子类的无参构造函数,无参构造函数如下:
有父类的: 多态! 先执行父类的无参构造!
父类的无参构造函数:
可以看出:创建了个beanFactory
然后走子类的无参构造函数:
第一个: 基于 注解方式
第二个: xml扫包方式
1走完了,继续2:
看下reader:
比如我们可以直接使用:
接着看2 : register()
继续看 register:
进行for循环遍历,多个注解情况
继续查看
查看该方法:注解情况的核心代码
public void registerBean(Class<?> annotatedClass, String name, Class... qualifiers) { // 基于注解的类: 作用是
AnnotatedGenericBeanDefinition abd = new AnnotatedGenericBeanDefinition(annotatedClass);
if (!this.conditionEvaluator.shouldSkip(abd.getMetadata())) {
ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(abd);
abd.setScope(scopeMetadata.getScopeName());
String beanName = name != null ? name : this.beanNameGenerator.generateBeanName(abd, this.registry);
AnnotationConfigUtils.processCommonDefinitionAnnotations(abd);
if (qualifiers != null) {
Class[] var7 = qualifiers;
int var8 = qualifiers.length; for(int var9 = 0; var9 < var8; ++var9) {
Class<? extends Annotation> qualifier = var7[var9];
if (Primary.class == qualifier) {
abd.setPrimary(true);
} else if (Lazy.class == qualifier) {
abd.setLazyInit(true);
} else {
abd.addQualifier(new AutowireCandidateQualifier(qualifier));
}
}
} BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(abd, beanName);
definitionHolder = AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);
BeanDefinitionReaderUtils.registerBeanDefinition(definitionHolder, this.registry);
}
}
传统的用: RootBeanDefinition 传统表示注入Bean对象
用注解的形式的话: AnnotatedGenericBeanDefinition
也是继承到了 BeanDefinition
也可以推理出 AnnotatedGenericBeanDefinition 表示为封装注解方式启动注入配置类。 注解方式启动的配置注入IOC Bean信息
根据条件注入Bean,Condition条件注入
设置回调
判断启动类上是否有Scope作用域,默认是单例的。设置作用域scope
beanName的默认和设置,没有就生成一个
调用了工具类
工具类要做的是: 读取配置类上面的 注解信息 比如有没有加上 @DependsOn 加载顺序 @Role 角色分类 @Primary 等
最终会在:registerBean方法中new
BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(abd, beanName);
通过AnnotatedGenericBeanDefinition 封装注解的启动类
DefinitionHolder 包装了 这个类: AnnotatedGenericBeanDefinition
然后通过工具类进行注册:
然后进行注册:
画图整理:
Spring5源码分析之AnnotationConfigApplicationContext的更多相关文章
- Spring5源码分析(1)设计思想与结构
1 源码地址(带有中文注解)git@github.com:yakax/spring-framework-5.0.2.RELEASE--.git Spring 的设计初衷其实就是为了简化我们的开发 基于 ...
- Spring5源码分析之Bean生命周期
Spring Bean生命周期的构成 Bean的完整生命周期经历了各种方法调用,这些方法可以划分为以下几类: Bean自身的方法: 这个包括了Bean本身调用的方法和通过配置文件中<bean&g ...
- Spring5源码分析之启动类的相关接口和注解
一些基础但是核心的知识总结: Spring Boot项目启动的时候需要加@Configuration. @ComponentScan @Configuration + @Bean 把第三方jar包注入 ...
- spring5源码分析系列(三)——IOC容器的初始化(一)
前言: IOC容器的初始化包括BeanDefinition的Resource定位.载入.注册三个基本过程. 本文以ApplicationContext为例讲解,XmlWebApplicationCon ...
- spring5源码分析系列(二)——spring核心容器体系结构
首先我们来认识下IOC和DI: IOC(Inversion of Control)控制反转:控制反转,就是把原先代码里面需要实现的对象创建.依赖的代码,反转给容器来帮忙实现.所以需要创建一个容器,并且 ...
- spring5源码分析系列(一)——spring5框架模块
spring总共大约20个模块,这些模块被整合在核心容器(Core Container).AOP和设备支持.数据访问及集成.Web.报文发送.Test 6个模块集合. 组成Spring框架的每个模块集 ...
- Spring5深度源码分析(三)之AnnotationConfigApplicationContext启动原理分析
代码地址:https://github.com/showkawa/spring-annotation/tree/master/src/main/java/com/brian AnnotationCon ...
- 5.2 Spring5源码--Spring AOP源码分析二
目标: 1. 什么是AOP, 什么是AspectJ 2. 什么是Spring AOP 3. Spring AOP注解版实现原理 4. Spring AOP切面原理解析 一. 认识AOP及其使用 详见博 ...
- 5.2 spring5源码--spring AOP源码分析二--切面的配置方式
目标: 1. 什么是AOP, 什么是AspectJ 2. 什么是Spring AOP 3. Spring AOP注解版实现原理 4. Spring AOP切面原理解析 一. 认识AOP及其使用 详见博 ...
随机推荐
- python算法与数据结构-选择排序算法(33)
一.选择排序的介绍 选择排序(Selection sort)是一种简单直观的排序算法.首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素, ...
- Caused by: java.io.IOException: 你的主机中的软件中止了一个已建立的连接。
org.apache.catalina.connector.ClientAbortException: java.io.IOException: 你的主机中的软件中止了一个已建立的连接. at org ...
- 关于跨域介绍和djiago解决跨域问题
什么是跨域? 跨域,指的是浏览器不能执行其他网站的脚本.它是由浏览器的同源策略造成的,是浏览器对javascript施加的安全限制. 什么是同源策略? 同源策略又分为以下两种 DOM同源策略:禁止对不 ...
- Backpack III
Description Given n kinds of items, and each kind of item has an infinite number available. The i-th ...
- Linux 使用shell脚本实现自动SSH互信功能
说假设有一个1000台节点的Hadoop集群,要配置节点之间的SSH免密码登录,该如何用shell脚本实现?#!/bin/expect #循环1000台机器的IP地址,生成密钥文件authorized ...
- HDP 企业级大数据平台
一 前言 阅读本文前需要掌握的知识: Linux基本原理和命令 Hadoop生态系统(包括HDFS,Spark的原理和安装命令) 由于Hadoop生态系统组件众多,导致大数据平台多节点的部署,监控极其 ...
- mysql 5.7 增删改查及别名的用法
1.启动和停止服务 一)启动和停止 #启动服务: $sudo service mysql start #停止服务: $sudo service mysql stop 二)创建和选择数据库 [创建数据库 ...
- OPCode详解及汇编与反汇编原理
1. 何为OPCode 在计算机科学领域中,操作码(Operation Code, OPCode)被用于描述机器语言指令中,指定要执行某种操作的那部分机器码,构成OPCode的指令格式和规范由处理器的 ...
- Ubuntu 系统安装教程
- Unity 绘制Mesh线条
using UnityEngine; using System.Collections; using System.Collections.Generic; public struct Segme ...