一、知识回顾
1.单例和多例
- 单例(Singleton)是SpringBoot的默认作用域,在整个Spring应用上下文中,单例作用域的Bean只会创建一个实例,并且所有对该Bean的请求都会返回这个唯一的实例
- 多例(Prototype)作用域表示每次从Spring容器中请求该Bean时,都会创建一个新的实例
二、面试实战
面试题1:Spring框架中的单例bean是线程安全的吗?
- 不是线程安全的
- Spring框架中有一个@Scope注解,默认的值就是Singleton,为单例的
- 因为一般在Spring的bean中注入的都是无状态对象(Service/Dao),无线程安全问题
- 如果在bean中定义了可修改的成员变量,是要考虑线程安全问题的,可以使用多例或者加锁来解决
面试题2:什么是AOP?你在项目中有没有使用到AOP?Spring中的事务是如何实现的?
- AOP称为面向切面编程,用于将那些与业务无关,但却对多个对象产生影响的公共行为和逻辑,抽取并封装为一个可重用的模块,这个模块被命名为“切面”(Aspect),减少系统中的重复代码,降低了模块间的耦合度,同时提高了系统的可维护性。
- 在项目中,常用的AOP使用场景为:记录操作日志,缓存处理,Spring中内置的事务处理,核心是:使用AOP中的环绕通知+切点表达式(找到要记录日志的方法),通过环绕通知的参数获取请求方法的参数(类,方法,注解,请求方式等),获取到这些参数以后,保存到数据库
- 其本质是通过AOP功能,对方法前后进行拦截,在执行方法之前开启事务,在执行完目标方法之后根据执行情况提交或者回滚事务
面试题3:Spring中事务失效的场景有哪些?
- 异常捕获处理:
原因:事务通知只有捕捉到了目标抛出的异常,才能进行后续的回滚处理,如果目标自己处理了异常,事务通知无法知悉
解决:在catch块添加throw new RuntimeException(e)抛出
- 抛出检查异常:
原因:Spring默认只会回滚非检查异常
解决:配置rollbackFor属性 @Transactional(rollbackFor = Exception.class)
- 非public方法导致的事务失效:
原因:Spring为方法创建代理,添加事务通知,前提条件都是该方法是public的
解决:改为public方法
面试题4:Spring的bean的生命周期
- 通过BeanDefinition获取bean的定义信息
- 调用构造函数实例化bean
- bean的依赖注入
- 处理Aware接口(BeanNameAware,BeanFactoryAware,ApplicationContextAware)
- Bean的后置处理器BeanPostProcessor-前置
- 初始化方法(InitializingBean,init-method)
- Bean的后置处理器BeanPostProcessor-后置
- 销毁bean
面试题5:Spring中的循环引用
- 循环依赖:两个或两个以上的bean互相持有对方,最终形成闭环,比如A依赖于B,B依赖于A
- 循环依赖在Spring中是允许存在的,Spring框架依据三级缓存已经解决了大部分的循环依赖:
- 一级缓存:单例池,缓存已经经历了完整的生命周期,已经初始化完成的bean对象
- 二级缓存:缓存早期的bean对象(生命周期还没走完)
- 三级缓存:缓存的是ObjectFactory,表示对象工厂,用来创建某个对象的
追问:构造方法出现了循环依赖怎么解决?
- A依赖于B,B依赖于A,注入的方式是构造函数
- 原因:由于bean的生命周期中构造函数是第一个执行的,Spring框架并不能解决构造函数的依赖注入
- 解决方案:使用@Lazy进行懒加载,什么时候需要对象再进行bean对象的创建
public A(@Lazy B b){System.out.println("A的构造方法执行了...");this.b = b; }
面试题6:SpringMVC的执行流程
面试题7:SpringBoot自动配置原理
- 在SpringBoot项目中的引导类上有一个注解@SpringBootApplication,这个注解对三个注解进行了封装,分别是:
- @SpringBootConfiguration,@EnableAutoConfiguration,@ComponentScan
- 其中@EnableAutoConfiguration是实现自动化配置的核心注解。该注解通过@Import注解导入对应的配置选择器。内部就是读取了该项目和该项目引用的Jar包的classpath路径下META-INF/spring.factories文件中的所配置的类的全类名。 在这些配置类中所定义的Bean会根据条件注解所指定的条件来决定是否需要将其导入到Spring容器中
- 条件判断会有像@ConditionalOnClass这样的注解,判断是否有对应的class文件,如果有则加载该类,把这个配置类的所有的Bean放入spring容器中使用
面试题8:Spring框架常见注解(Spring,SpringBoot,SpringMVC)
面试题9:MyBatis执行流程
面试题10:MyBatis是否支持延迟加载?
- 延迟加载:在需要用到数据时才进行加载,不需要用到数据时就不加载数据
- MyBatis支持一对一关联对象和一对多关联集合对象的延迟加载
- 在MyBatis配置文件中,可以配置是否启用延迟加载lazyLoadingEnabled = true/false,默认是关闭的
追问:延迟加载的底层原理?
面试题11:MyBatis的一级、二级缓存用过吗?
- 一级缓存:基于 PerpetualCache 的 HashMap 本地缓存,其存储作用域为 Session,当Session进行flush或close之后,该Session中的所有Cache就将清空,默认打开一级缓存
- 二级缓存是基于namespace和mapper的作用域起作用的,不是依赖于SQL session,默认也是采用PerpetualCache,HashMap 存储。需要单独开启,一个是核心配置,一个是mapper映射文件
追问:MyBatis的二级缓存什么时候会清理缓存中的数据?
- 当某一个作用域(一级缓存 Session/二级缓存Namespaces)的进行了新增、修改、删除操作后,默认该作用域下所有 select 中的缓存将被 clear