一、前言
在之前我们使用文字分析过Spring的几个核心原理(IOC, DI. AOP),本章会把Spring中的Bean生命周期进行较为详细的讲解
Spring相关文章汇总(Ps:有部分还未完成):
- Spring中IOC DI和AOP分别是什么
- Bean的声明周期底层原理
- 依赖注入底层原理
- 初始化底层原理
- 构造方法底层原理
- AOP底层原理
二、Bean的创建生命周期
- 通过推断构造方法来实例化一个对象
- 对该对象进行依赖注入(属性赋值)
- 依赖注入后,Spring会判断该对象是否实现了BeanNameAware接口、BeanClassLoaderAware接口、BeanFactoryAware接口,如果实现了,就表示当前对象必须实现该接口中所定义的setBeanName()、setBeanClassLoader()、setBeanFactory()方法,那Spring就会调用这些方法并传入相应的参数(Aware回调)
- 初始化前(@PostConstruct)
- 初始化(继承InitializingBean类)
- 初始化后(AOP)
- 获取到代理对象
三、Spring中是通过什么方式创建对象的
在java语言中肯定是通过某个类去创建对象的, 下面是实例代码
public static void main(String[] args) { // 创建一个Spring对象 AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class); // 创建对象 NingxuanService ningxuanService = (NingxuanService)context.getBean("ningxuanService"); // 执行方法 ningxuanService.test(); } 复制代码
一共分为三步走:
- 解析Appconfig类, 获取扫描路径, 获取Bean对象
- 扫描包下的所有类文件的注解, 如果发现 @Component, @Service, @Bean等注解的时候就会将这些注解给记录下来, 存到一个Map中
- Spring会根据相应的规则生成对应的BeanName作为map的key, value是当前类
这个时候调用 context.getBean("BeanName")的时候就会获取到对应的实体类, 从而去创建对象
三、Bean生命周期详解
首先, 我们新建一个Config类, 设定好包扫描路径同时开启AOP
@ComponentScan("com.ningxuan") @EnableAspectJAutoProxy public class AppConfig { @Bean public NingxuanService ningxuanService(){ return new NingxuanService(); } } 复制代码
推断构造方法
Spring在基于某个类生成Bean的时候, 会通过该类的构造方法去实例化一个对象, 但是如果这个类拥有多个构造方法的时候就会进入以下的判断
- 只有一个构造方法, 那么不管这个构造犯法是有参的还是无参的都会使用这个构造方法
- 如果存在多个构造方法
- 在多个构造犯法中, 若存在无参构造方法, 则默认使用无参构造犯法
- 若该有参构造方法上有使用 @Autowired注解, 则使用有参构造
- 若多个构造方法皆为有参构造
- 若构造方法上都没有加 @Autowired注解, 则报错
- 若有某一个构造方法上加了 @Autowired注解, 则使用该构造方法
- 若存在多个构造方法上交了 @Autowired注解, 则报错
Spring会根据入参的类型和名字去Spring中找Bean对象:
- 先根据入参类型找, 若只找到一个, 那就直接用来作为入参
- 如果根据类型找到多个, 则在根据入参名字来确定唯一的一个
- 最终如果没有找到, 则报错
确定用哪个构造方法, 确定入参的Bean对象, 这个古城就叫做推断构造方法
依赖注入
通过推断构造方法得到一个对象后, Spring会判断该对象中是否存在被 @Autowired 注解了的属性, 把这些属性找出来并由Spring进行赋值
初始化前
被 @PostConstruct 注解标注的方法, 是为该bean初始化前执行的方法, 具体如下
@Component() public class NingxuanService{ // 初始化前执行 @PostConstruct public void after(){ System.out.println("初始化前"); } public void test(){ System.out.println("执行方法: 宁轩"); } } 复制代码
初始化
想要一个方法在被Spring创建对象初始化时执行, 则要将这个类继承 InitializingBean 类, 并实现 afterPropertiesSet 方法, 且要抛出异常, 具体如下
@Component() public class NingxuanService implements InitializingBean { // 初始化前执行 @PostConstruct public void after(){ System.out.println("初始化前"); } public void test(){ System.out.println("执行方法: 宁轩"); } // 初始化执行 @Override public void afterPropertiesSet() throws Exception { System.out.println("初始化"); } } 复制代码
AOP
想要执行AOP, 老规矩, 在扫描类上使用注解 @EnableAspectJAutoProxy 进行标注, , 并且新建一个aop类, 使用 @Aspect和@Component 注解标注, 本次测试采用了 AOP 中的前置执行和后置执行, 具体如下:
@Aspect @Component public class TestAop { @After("execution(public void com.ningxuan.demo.service.NingxuanService.test())") public void TestAfter(JoinPoint joinPoint){ System.out.println("方法执行后"); } @Before("execution(public void com.ningxuan.demo.service.NingxuanService.test())") public void TestVefore(JoinPoint joinPoint){ System.out.println("方法执行前"); } } 复制代码
执行测试
根据以上的代码, 我们执行 main 方法, 可以得到以下测试结果
网络异常,图片无法展示
|