图片来源: https://zhuanlan.zhihu.com/p/38208324
尚硅谷视频地址:https://www.bilibili.com/video/BV1gW411W7wy
核心思想:
1、Spring容器在启动的时候,先会保存所有注册进来的Bean的定义信息;
- xml注册bean
- 注解注册Bean;@Service、@Component、@Bean、xxx
2、Spring容器会合适的时机创建这些Bean
- 用到这个bean的时候;利用getBean创建bean;创建好以后保存在容器中;
- 统一创建剩下所有的bean的时候;finishBeanFactoryInitialization();
3、后置处理器;BeanPostProcessor
- 每一个bean创建完成,都会使用各种后置处理器进行处理;来增强bean的功能;
- AutowiredAnnotationBeanPostProcessor:处理自动注入
- AnnotationAwareAspectJAutoProxyCreator:来做AOP功能;
4、事件驱动模型
- ApplicationListener:事件监听;
- ApplicationEventMulticaster:事件派发:
Spring IOC
@Configuration
指定是一个配置类,自身也会被加入容器中
@Bean
添加组件
@ComponentScan
指定要扫描得包,还可以指定包含哪些组件、不包含哪些组件
可通过实现TypeFilter来自定义过滤器,:@ComponentScan.Filter(type = FilterType.CUSTOM, classes = {MyTypeFilter.class})
,其中MyTypeFilter是TypeFilter实现类
@Scope
设置单实例(默认)、多实例
- ConfigurableBeanFactory.SCOPE_PROTOTYPE,多例
- ConfigurableBeanFactory.SCOPE_SINGLETON,单例
- org.springframework.web.context.WebApplicationContext.SCOPE_REQUEST,同一请求创建一个实例
- org.springframework.web.context.WebApplicationContext.SCOPE_SESSION, 同一session创建一个实例
@Lazy
在容器启动时不创建,在第一次获取Bean时创建对象并初始化。
@Conditional
传入一个Condition数组,Condition接口:
@FunctionalInterface
public interface Condition {
/**
* ConditionContext:判断条件能使用的上下文环境
* AnnotatedTypeMetadata:注释信息
*/
boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata);
}
组件注册方法
@Controller、@Service、@Repository、@Component
包扫描 + 组件标注注解(@Controller、@Service、@Repository、@Component),这种方式只能注册自己写的类
@Bean
@Bean,可以注册第三方包的类
@Import
@Import:快速给容器中导入一个组件,id默认是全类名,除了可以导入一般的类,还可以导入以下内容用于注册组件:
-
ImportSelector:一个接口,返回需要导入的组件的全类名数组
- 实现selectImports(AnnotationMetadata importingClassMetadata)方法
- 参数importingClassMetadata:当前标注@Import注解类的所有信息
- 实现selectImports(AnnotationMetadata importingClassMetadata)方法
-
ImportBeanDefinitionRegistrar:一个接口,手动注册bean到容器中,可以自定义组件名称
- 实现registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry)方法
- 参数importingClassMetadata:当前标注@Import注解类的所有信息
- 参数registry注册组件的类,可用于手动注册组件
- 通过registry.registerBeanDefinition方法注册组件
- 实现registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry)方法
FactoryBean
`使用Spring提供的FactoryBean(工厂Bean),getObject()添加组件,还可以定义组件类型、是否单例
Bean生命周期
描述
bean从创建 ----> 初始化 -----> 销毁的过程
容器管理bean的生命周期:
1、对象创建
- 单实例:在容器启动的时候创建对象
- 多实例:在每次获取的时候创建对象
2、初始化
- 对象完成创建,并赋值好后,调用初始化方法
3、销毁
- 单实例:在容器关闭的时候销毁
- 多实例:容器不会管理这个bean,所以不会调用销毁方法
自定义组件初始化和销毁方法
我们可以自定义初始化和销毁方法
初始化和销毁方法:
- 通过
@Bean
注解指定init-method和destroy-method - 实现
InitializingBean
和DisposableBean
接口,重写里面的destroy和afterPropertiesSet方法 - 使用
@PostConstruct
和@PreDestroy
注解- PostConstruct:在bean创建完成并且属性赋值完成后进行初始化
- PreDestroy:在容器销毁bean之前执行
BeanPostProcessor
:接口,bean后置处理器,在bean初始化前后进行一些处理- postProcessBeforeInitialization:在初始化之前执行
- postProcessAfterInitialization:在初始化之后执行
BeanPostProcessor执行过程
- 执行populateBean(beanName, mbd, instanceWrapper);给bean进行属性赋值
- 开始initializeBean初始化bean
- 先执行applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName);,遍历得到容器中所有的BeanPostProcessor;挨个执行beforeInitialization,一但返回null,跳出for循环,不会执行后面的BeanPostProcessor.postProcessorsBeforeInitialization
- 然后执行invokeInitMethods(beanName, wrappedBean, mbd);执行自定义初始化
- 最后执行applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName);
@Value
使用@Value赋值,赋值方法:
- 直接在@Value中写基本数值
- 使用SpEL表达式:#{},如#{6+6}表示12
- 使用${}:取出配置文件中的值(在运行环境变量里的值)
@PropertySource
例如:@PropertySource(“classpath:a.properties”),可以读取a.properties中的k-v值,之后使用${...}
可以取出值
@AutoWired、@Resource、@Inject
都可以给属性注入值
1、@AutoWired
-
默认优先按照类型去容器中找对应的组件,找到就赋值。
-
如果该类型的组件有多个,再将属性名作为组件的id去容器中查找
-
required属性:当required=false时,表示这个组件不是必须的
-
@Qualifier需要和@AutoWired一起使用,用于指定id
-
@Primary指定当前bean是首选的
-
可以在构造器、参数、方法、属性上标注
-
我们可以在有参构造函数上标注@Autowired,让IOC容器创建组件时调用该类的有参构造方法
- 默认加在IOC容器中的组件,容器启动会调用无参构造器创建对象,再进行初始化赋值等操作;
- 如果组件只有一个有参构造器,这个有参构造器的@Autowired可以省略
-
标在参数上,自定义类型的值从IOC容器中获取
-
标注在方法上,Spring容器创建当前对象,就会调用方法,完成赋值
-
标在属性上,会从ioc容器中获取并赋值
-
2、@Resource
- 可以和@Autowired一样实现自动装配功能,但默认是按组件名称进行装配
- 可以给属性name赋值,自定义组件名称
3、@Inject
-
需要导入javax.inject依赖
-
和@Autowired功能一样,但没有required=false的功能
使用Spring容器底层组件(实现xxxAware)
自定义组件想要使用Spring容器底层的一些组件(ApplicationContext、BeanFactory),需要自定义组件实现xxxAware接口
在创建对象的时候,会调用接口规定的方法,注入相关的组件
例如:
自定义类实现ServletContextAware接口,于是我就能拿到ServletContext进行操作
@Profile
1、@Profile注解是Spring为我们提供的可以根据当前环境,动态的激活和切换一系列组件的功能
- 可以写在类上和方法上
- 例如:@Profile(“test”),只有在运行环境时test的时候,组件才会生效
- 没有添加@Profile注解,表示任何环境下都生效
2、指定运行环境的方式:
-
命令行方式,启动时指定参数:
-Dspring.profiles.active=test
-
代码方式步骤:
- 1、使用无参构造器创建application,AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext();
- 2、设置需要激活的环境:applicationContext.getEnvironment().setActiveProfiles(“test”, “dev”);
- 3、注册主配置类:applicationContext.register(MainConfig11.class);
- 4、刷新容器:applicationContext.refresh();
SpringAOP
- 【动态代理】
- 指在程序运行期间动态将某段代码切入到指定方法的指定玩位置进行运行的编程方式
步骤
使用步骤
-
导入aop模块;Spring AOP:(spring-aspects)
<dependency> <groupId>org.springframework</groupId> <artifactId>spring-aspects</artifactId> </dependency>
-
业务逻辑类
-
切面类
-
给切面类的目标方法标注解
- 前置通知(
@Before
):在目标方法运行之前运行 - 后置通知(@After):在目标方法运行结束之后运行(无论方法正常结束还是异常结束)
- 返回通知(
@AfterReturning
):在目标方法正常返回之后运行 - 异常通知(
@AfterThrowing
):在目标方法出现异常以后运行 - 环绕通知(
@Around
):动态代理,手动推进目标方法运行(joinPoint.procced())
- 前置通知(
-
将切面类和业务逻辑类(目标方法所在类)都`加入到容器中
-
必须告诉Spring哪个类是切面类(给切面类上加一个注解:
@Aspect
) -
给配置类中加
@EnableAspectJAutoProxy
【开启基于注解的aop模式】.
注意:JoinPoint 一定要作为第一个参数
例子
- 1、引入依赖
- 2、业务逻辑类,例如定义一个controller:
- 3、定义切面类、标注解、声明这是切面类、开启注解对aop的支持,例如:
execution表达式
参考于:
- https://blog.csdn.net/qq_41460654/article/details/119637891
- https://blog.csdn.net/somilong/article/details/74568223
表达式的结构
访问修饰符[可省] 返回值 包名.包名.包名...类名.方法名(参数列表)
其他说明
-
返回值可以使用通配符 * ,表示任意返回值
-
包名可以使用通配符 * ,表示任意包。但是有几级包,就需要写几个*
-
是包和包、包和类等之间的连接符
-
包名可以使用 … 表示当前包及其子包
-
类名和方法名都可以使用*来实现通配
-
参数列表:
- 可以直接写数据类型:
- 基本类型直接写名称,例如:int
- 引用类型写包名.类名的方式,例如:java.lang.String
- 可以使用通配符 * 表示任意类型,但是必须有参数
- 可以使用… 表示有无参数均可,有参数可以是任意类型
- 可以直接写数据类型:
-
全通配写法:(表示匹配所有包下的所有类的所有方法):
* *..*.*(..)
-
通常写法:
切到业务层实现类下的所有方法:* com.ljy.service.impl.*.*(..)
AnnotationAwareAspectJAutoProxyCreator流程图
本节图片来源:https://blog.csdn.net/xjhqre/article/details/123264069
注册AnnotationAwareAspectJAutoProxyCreator的后置处理器流程图:
AnnotationAwareAspectJAutoProxyCreator执行时机流程图:
AnnotationAwareAspectJAutoProxyCreator配置代理流程图:
声明式事务 (略)
视频地址:https://www.bilibili.com/video/BV1gW411W7wy?p=38
基本原理:
- 执行目标方法,如果异常,获取到事务管理器,利用事务管理回滚操作;
- 如果正常,利用事务管理器,提交事务
底层类
1、BeanPostProcessor
bean后置处理器,bean创建对象初始化前后进行拦截工作的;主要有以下两个方法:
- postProcessBeforeInitialization,在bean初始化之前执行
- postProcessAfterInitialization,在bean初始化之后执行
2、BeanFactoryPostProcessor
beanFactory的后置处理器:
- 在BeanFactory标准初始化之后调用,来定制和修改BeanFactory的内容
- 所有的bean定义已经保存加载到beanFactory,但是bean的实例还未创建
3、BeanDefinitionRegistryPostProcessor
它extends BeanFactoryPostProcessor
-
其优先于BeanFactoryPostProcessor执行;在所有bean定义信息将要被加载,bean实例还未创建的时候执行
-
可以利用BeanDefinitionRegistryPostProcessor给容器中再额外添加一些组件
4、ApplicationListener
ApplicationListener<E extends ApplicationEvent>
extends EventListener
- 该类有一个方法:onApplicationEvent,当事件发生后执行
- 可以监听自定义事件,需要自定义事件,然后可调用applicationContext.publishEvent()发布事件
自定义事件示例:
-
定义事件,继承ApplicationEvent
-
定义监听器,监听自定义事件
-
发布自定义事件:
- 另外,个人觉得可以把run作设置为静态成员变量,以便其他地方发布事件,或者实现
ApplicationContextAware
接口,也可以获得ApplicationContext
- 另外,个人觉得可以把run作设置为静态成员变量,以便其他地方发布事件,或者实现
-
结果: