Spring底层原理大致脉络

目录

Spring是如何创建一个对象的

Bean的创建过程

AOP的大致流程


Spring是如何创建一个对象的

其实不管是AnnotationConfigApplicationContext还是 ClassPathXmlApplicationContext,目前,我们都可以简单的将它们理解为就是用来创建 Java对象的,比如调用getBean()就会去创建对象。

java 复制代码
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
UserService userservice = (UserService) context.getBean("UserService");
userservice.login();

当我们调用context.getBean("userService")时,就会去创建一个对象,但是getBean方法 内部怎么知道"userService"对应的是UserService类呢? 所以,我们就可以分析出来,在调用AnnotationConfigApplicationContext的构造方法 时,也就是第一行代码,会去做一些事情:

  1. 解析AppConfig.class,得到扫描路径

  2. 遍历扫描路径下的所有Java类,如果发现某个类上存在@Component、 @Service等注解,那么Spring就把这个类记录下来,存在一个Map中,比如 Map。(实际上,Spring源码中确实存在类似的这么一个Map,叫 做BeanDefinitionMap)

  3. Spring会根据某个规则生成当前类对应的beanName,作为key存入Map,当前 类作为value 这样,但调用context.getBean("userService")时,就可以根据"userService"找到 UserService类,从而就可以去创建对象了。

Bean的创建过程

那么Spring到底是如何来创建一个Bean的呢,这个就是Bean创建的生命周期,大致过程如下

  1. 利用该类的构造方法来实例化得到一个对象(但是如何一个类中有多个构造方法, Spring则会进行选择,这个叫做推断构造方法)

  2. 得到一个对象后,Spring会判断该对象中是否存在被@Autowired注解了的属 性,把这些属性找出来并由Spring进行赋值(依赖注入)

  3. 依赖注入后,Spring会判断该对象是否实现了BeanNameAware接口、 BeanClassLoaderAware接口、BeanFactoryAware接口,如果实现了,就表示当前 对象必须实现该接口中所定义的setBeanName()、setBeanClassLoader()、 setBeanFactory()方法,那Spring就会调用这些方法并传入相应的参数(Aware回 调)

  4. Aware回调后,Spring会判断该对象中是否存在某个方法被@PostConstruct注解 了,如果存在,Spring会调用当前对象的此方法(初始化前)

  5. 紧接着,Spring会判断该对象是否实现了InitializingBean接口,如果实现了,就 表示当前对象必须实现该接口中的afterPropertiesSet()方法,那Spring就会调用当前对象中的afterPropertiesSet()方法(初始化)

  6. 最后,Spring会判断当前对象需不需要进行AOP,如果不需要那么Bean就创建完 了,如果需要进行AOP,则会进行动态代理并生成一个代理对象做为Bean(初始化 后)

通过最后一步,我们可以发现,当Spring根据UserService类来创建一个Bean时:

  1. 如果不用进行AOP,那么Bean就是UserService类的构造方法所得到的对象。

  2. 如果需要进行AOP,那么Bean就是UserService的代理类所实例化得到的对象,而 不是UserService本身所得到的对象。

Bean对象创建出来后:

  1. 如果当前Bean是单例Bean,那么会把该Bean对象存入一个Map,Map的key为beanName,value为Bean对象。这样下次getBean时就可 以直接从Map中拿到对应的Bean对象了。(实际上,在Spring源码中,这个Map就 是单例池)

  2. 如果当前Bean是原型Bean,那么后续没有其他动作,不会存入一个Map,下次 getBean时会再次执行上述创建过程,得到一个新的Bean对象。

AOP的大致流程

AOP就是进行动态代理,在创建一个Bean的过程中,Spring在最后一步会去判断当前正在创建的这个Bean是不是需要进行AOP,如果需要则会进行动态代理。

如何判断当前Bean对象需不需要进行AOP:

  1. 找出所有的切面Bean

  2. 遍历切面中的每个方法,看是否写了@Before、@After等注解

  3. 如果写了,则判断所对应的Pointcut是否和当前Bean对象的类是否匹配

  4. 如果匹配则表示当前Bean对象有匹配的的Pointcut,表示需要进行AOP

利用cglib进行AOP的大致流程:

  1. 生成代理类UserServiceProxy,代理类继承UserService

  2. 代理类中重写了父类的方法,比如UserService中的test()方法

  3. 代理类中还会有一个target属性,该属性的值为被代理对象(也就是通过 UserService类推断构造方法实例化出来的对象,进行了依赖注入、初始化等步骤的 对象)

  4. 代理类中的test()方法被执行时的逻辑如下:

a. 执行切面逻辑(@Before)

b. 调用target.login()

当我们从Spring容器得到UserService的Bean对象时,拿到的就是UserServiceProxy所生 成的对象,也就是代理对象。 UserService代理对象.test()--->执行切面逻辑--->target.test(),注意target对象不是代理 对象,而是被代理对象。

相关推荐
带刺的坐椅5 分钟前
超越 SpringBoot 4.0了吗?OpenSolon v3.8, v3.7.4, v3.6.7 发布
java·ai·springboot·web·solon·flow·mcp
就叫飞六吧5 分钟前
基于spring web实现简单分片上传demo
java·前端·spring
apihz11 分钟前
免费手机号归属地查询API接口详细教程
android·java·运维·服务器·开发语言
程序员小假13 分钟前
学院本大二混子终于找到实习了...
java·后端
回吐泡泡oO14 分钟前
找不到rar.RarArchiveInputStream?JAVA解压RAR5的方案。
java·开发语言
jiayong2316 分钟前
AI应用领域编程语言选择指南:Java vs Python vs Go
java·人工智能·python
bjzhang7520 分钟前
IDEA 2025.3重磅发布,Ultimate 终极版和 Community社区版二合一,免费版可商用
java·idea
武子康20 分钟前
大数据-194 数据挖掘 从红酒分类到机器学习全景:监督/无监督/强化学习、特征空间与过拟合一次讲透
大数据·后端·机器学习
程序猿零零漆22 分钟前
Spring之旅 - 记录学习 Spring 框架的过程和经验(三)Bean的依赖注入配置、Spring的其它配置标签
java·学习·spring
TT哇25 分钟前
@AllArgsConstructor
java·开发语言