继续,第四波女朋友面试4个月的面试题复盘,这次是关于Spring的题目。
Spring是Java web开发接触最多的一个框架,基本上每天都会打交道。
直接上题目。Spring框架虽然有很多组件,但是面试题目还是比较少的,因为他的核心思想比较集中。还是老套路,高频的面试题已经标星,有面试需要的同学可以先点星星收藏起来。
Spring的源码非常深,Spring框架使用了各种设计模式,因此阅读Spring的源码比一些国产框架源码会吃力一些,但是这也阻挡不了我们了解Spring的原理之路,只要我们平时对Spring的原理有一定了解,我们也可以讲清楚常见的面试题目。
下面开始进入面试题的复盘。
对Spring的思考
- Spring的好处
这种开放性问题,没有标准的答案,我们不要等面试官问完就回答,我们可以先思考2秒,脑袋里将Spring进行总结一下。我们按照金字塔原理的方式回答,记住一定要先说重点。
1、Spring通过Ioc和Di技术为Bean提供了生命周期管理能力,帮助程序员创建bean和管理bean的关系
2、Spring提供了面向切面编程Aop的编程方式,可以帮助程序员快速集成通用的非业务功能,比如内置的事务功能就是通过aop实现的。
3、Spring框架提供了非常多的扩展点,帮助程序员快速开发和集成第三方组件,比如bean的处理器BeanPostProcessor
,全局异常处理ControllerAdvice
切面等。
有些面试官可能会根据你说的发散的问,所以你回答的点最好都是自己比较熟悉的点。
Ioc模块面试题
- 说说Bean的生命周期
如果了解Spring创建Bean的过程,那么这个题目回答起来就没问题。
先说核心重点
再详细说明
首先bean的出生就是Spring先拿到bean的配方BeanDefinition
信息,然后通过反射技术实例化一个bean对象出来。
这个时候bean只是创建了实例,他的属性都是默认值,没有进行用户定制化的属性填充,所以第二步是属性填充。
属性填充之后,Spring为了给用户一些定制化bean的机会,给每个bean留下了一些初始化的接口。
aware后缀接口,感知Spring框架bean的能力
bean的后置处理器,初始化前回调函数
org.springframework.beans.factory.config.BeanPostProcessor#postProcessBeforeInitialization
invokeInitMethods 初始化方法回调
InitializingBean.afterPropertiesSet
invokeCustomInitMethod 包含:PostConstruct注解
- bean的后置处理器,初始化后回调函数
org.springframework.beans.factory.config.BeanPostProcessor#postProcessAfterInitialization
初始化完成之后Bean就可以被业务程序使用了,直到程序销毁的时候,bean也会随之销毁。
- Spring循环依赖的解决方法
Spring解决循环依赖使用到了三个map作为三级缓存,分别存储不同状态的bean。
/** 一级缓存 存储可以直接使用的bean **/
private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);
/** 二级缓存 存储bean工厂 可以延迟调用实例化 使用的时候才调用工厂方法获取对象 */
private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>(16);
/** 存储不完整对象 不能直接用的 循环依赖时,某些依赖的属性没有设置,这个时候bean就是不完整对象 */
private final Map<String, Object> earlySingletonObjects = new HashMap<>(16);
总结一下,为什么使用三级缓存?
1、singletonObjects这个缓存是存储完整的对象,可以直接使用的。
2、singletonFactories这个缓存是为了延迟初始化,是解决循环依赖的关键,存在循环依赖注入时,可能注入的对象需要被代理,这个工厂类来延迟实例化一个代理类。
3、earlySingletonObjects这个缓存是为了在A注入B,而B又注入A这个阶段,A是一个半成品,需要用来存储A这种半成品bean的状态。需要等待A将B注入之后,这个缓存就要移除。
4、singletonFactories和earlySingletonObjects都是存储对象的中间状态,依赖它们保证最终注入的对象是完整的,依赖注入完成后会被清除。
aop模块
- 说说aop的原理
Spring aop是对动态代理技术的封装组件,帮助开发人员使用aop实现可插拔的切面功能。
我们可以分三个阶段和面试官说明。
第一阶段 解析程序中定义的切面
这个阶段是解析aop注解或者标签的过程,比如Aspect标识的类,解析得到aop定义的基础beanAspectJPointcutAdvisor
,Advice
第二阶段 查找方法的切面和通知,构建通知
对于需要被切面拦截的bean,在bean实例化之后,开始判断是否要创建代理对象,通过jdk或者cglib,同时构建了通知链数组,并缓存起来。
使用方法对象和方法hashCode作为key
private final Method method;
private final int hashCode;
第三阶段 调用通知链路,执行切面的逻辑。
如果是jdk动态代理调用目标方法会触发
org.springframework.aop.framework.JdkDynamicAopProxy#invoke
方法
如果是cglib,调用目标方法会触发CglibAopProxy中
org.springframework.aop.framework.CglibAopProxy.DynamicAdvisedInterceptor#intercept
方法
Spring boot模块面试题
- 自动装配的原理
其实自动装配是Springboot借助Spring提供导入bean到Spring容器的一些机制来实现的。
Spring提供了@Import
,ImportSelector
,DeferredImportSelector
, ImportBeanDefinitionRegistrar
等方式导入bean到Spring容器中,
同时利用spring spi技术,查找配置在spring.factories
文件里的自动配置类,这样我们只要按照SpringBoot定义规范编写需要自动装配的类就可以实现自动装配了。
总结
Spring和Springboot的题目想回答的时候更加有自信心,对于ioc,aop以及他们的扩展点的原理一定要比较清晰,Spring其他能力都是在这两个核心能力上构建的。