1、Spring框架中的单例bean是线程安全的吗?
不是线程安全的,Spring框架中有一个@Scope注解,默认的值就是singleton,单例的,
因为一般在Spring的bean的中都是注入无状态的对象,没有线程安全问题,如果在bean中定义了可修改的成员变量,是要考虑线程安全问题的,可以使用多例或者加锁来解决,
比如:我们通常在项目中使用的Spring bean都是不可变的状态(比如Service类和DAO类),所以在某种程度上来说Spring的单例bean是线程安全的
如果你的bean有多种状态的话(比如View Model对象),就需要自行保证线程安全,最浅显的解决办法就是将多态bean的作用由“singleton”变更为“prototype”
2、什么是AOP?
AOP称为面向切面编程,用于讲那些与业务无关,但却对多个对象产生影响的公共行为和逻辑,抽取并封装为一个可重用的模块,这个模块被命名为“切面”(Aspect),减少系统中的重复代码,降低了模块间的耦合度,同时提高了系统的可维护性。
3、项目中有没有使用到AOP?
记录操作日志,缓存,Spring实现的事务
核心是:使用AOP中的环绕通知 + 切点表达式(找到要记录日志的方法),通过环绕通知的参数获取请求方法的参数(类、方法、注解、请求方式等),获取到这些参数以后,保存到数据库
4、Spring中的事务是如何实现的?
Spring支持编程式事务管理和声明式事务管理两种方式。
- 编程式事务管理:需要使用TransactionTemplate来进行实现,对业务代码有侵入性,项目中很少使用
- 声明式事务管理:声明式事务管理建立在AOP之上的,其本质是通过AOP功能,对方法前后进行拦截,将事务处理的功能编织到拦截的方法中,也就是在目标方法开始之前加入一个事务,在执行完目标方法之后根据执行情况提交事务或者回滚事务。用到的注解@Transactional
5、Spring中事务失效的场景有哪些?
1)异常捕获处理
- 自己处理了异常,并没有抛出异常,解决方法就是手动抛出异常即可
2)抛出检查异常
- Spring默认只回滚非检查异常,解决方法就是在@Transactional配置rollbackFor属性为Exception
3)非public方法导致的事务失效
- Spring为方法创建代理,添加事务通知,前提条件都是该方法是public的,解决方法是改为public修饰符即可
6、Spring的bean的生命周期?
- 通过BeanDefinition获取bean的定义信息
- 调用构造函数实例化bean
- bean的依赖注入,一般是set方法注入
- 处理Aware接口,(BeanNameAware、BeanFactoryAware,ApplicationContextAware)
- Bean的后置处理器BeanPostProcessor-前置
- 初始化方法(initializingBean、init-method)
- Bean的后置处理器BeanPostProcessor-后置
- 销毁bean
7、Spring中的循环引用?
- 循环依赖:循环依赖其实就是循环引用,也就是两个或者两个以上的bean互相持有对方,最终形成闭环,比如A依赖B,B依赖A
- 循环依赖在Spring中是允许存在的,Spring框架依据三级缓存已经解决了大部分的循环依赖
①、一级缓存:单例池,缓存已经经历了完整的生命周期,已经初始化完成的bean对象
②、二级缓存:缓存早期的bean对象(生命周期还没走完)
③、三级缓存:缓存的是ObjectFactory,表示对象工厂,用来创建某个对象的
8、构造方法出现了循环依赖怎么解决?
A依赖于B,B依赖于A,注入的方式是构造函数
原因:由于bean的生命周期中构造函数是第一个执行的,spring框架并不能解决构造函数的依赖注入
解决方法:使用@Lazy进行懒加载,什么时候需要对象再进行bean对象的创建
9、SpringMVC的执行流程知道吗?(视图版本,JSP)
1)用户发出请求到前端控制器DispatcherServlet
2)DispatcherServlet收到请求调用HandlerMapping(处理器映射器)
3)HandlerMapping找到具体的处理器,生成处理器对象及处理器拦截器(如果有),再一起返回给DispatcherServlet
4)DispatcherServlet调用HandlerAdapter(处理器适配器)
5)HandlerAdapter经过适配调用具体的处理器(Handler/Controller)
6)Controller执行完成返回ModelAndView对象
7)HandlerAdapter将Controller执行结果ModelAndView返回给DispatcherServlet
8)DispatcherServlet将ModelAndView传给ViewReslover(视图解析器)
9)ViewReslover解析后根据具体View(视图)
10)DispatcherServlet根据View进行渲染视图(即将模型数据填充至视图中)
11)DispatcherServlet响应用户
9、SpringMVC的执行流程知道吗?(JSON版本,前后端分离)
1)用户发出请求到前端控制器DIspatcherServlet
2)DispatcherServlet收到请求调用HandlerMapping(处理器映射器)
3)HandlerMapping找到具体的处理器,生成处理器对象及处理器拦截器(如果有),再一起返回给DispatcherServlet
4)DispatcherServlet调用HandlerAdapter(处理器适配器)
5)HandlerAdapter经过适配调用具体的处理器(Handler/Controller)
6)方法上添加了@ResponseBody
7)通过HttpMessageConverter来返回结果转换为JSON并响应
10、SpringBoot的自动配置原理(重点)
1)在SpringBoot项目中的引导类上有一个注解@SpringBootApplication,这个注解是对三个注解进行了封装,分别是:
- @SpringBootConfiguration:与@Configuration注解作用相同,用来声明当前也是一个配置类
- @ComponentScan:组件扫描,默认扫描当前引导类所在包及其子包
- @EnableAutoConfiguration:SpringBoot实现自动化配置的核心注解
2)其中@EnableAutoConfiguration是实现自动化配置的核心注解,该注解通过@Import注解导入对应的配置选择器,内部就是读取了该项目和该项目引用的Jar包的classpath路径下的META-INF/spring.factoties文件中所配置的类的全类名,在这些配置类中所定义的Bean会根据条件注解所指定的条件来决定是否需要将其导入到Spring容器中
3)条件判断会有像@ConditionOnClass这样的注解,判断是否有对应的class文件,如果有则加载该类,把这个配置类的所有Bean放入Spring容器中使用。
11、Spring中常用的注解有哪些?
12、SpringMVC中常用的注解有哪些?
13、SpringBoot中常用的注解有哪些?
14、Mybatis执行流程
1)读取Mybatis核心配置文件:mybatis-config.xml加载运行环境和映射文件
2)构造会话工厂SqlSessionFactory,只能有一个
3)会话工厂创建SqlSession对象(包含了执行sql语句的所有方法)可以有多个
4)操作数据库的接口,Executor执行器,同时负责查询缓存的维护
5)Executor接口的执行方法中有一个MappendStatement类型的参数,封装了映射信息
6)输入参数映射(就是将java的参数映射为数据库的参数)
7)输入结果映射(就是将数据库的参数映射为java的参数)
15、Mybatis是否支持延迟加载?
- 延迟加载的意思是:就是在需要用到数据时才进行加载,不需要用到数据时就不加载数据
- Mybatis支持一对一关联对象和一对多关联集合对象的延迟加载
- 在Mybatis配置文件中,可以配置是否启用延迟加载lazyLoadingEnabled=true|false,默认是关闭的
16、延迟加载的底层原理是什么?
1)使用CGLIB创建目标对象的代理对象
2)当调用目标方法时,进入拦截器invoke方法,发现目标方法时null值,执行sql查询
3)获取数据以后,调用set方法设置属性值,再继续查询目标方法,就有值了
17、Mybatis的一级,二级缓存用过吗?
- 一级缓存:基于PerpetualCache的HashMap本地缓存,其存储作用域为Session,当Session进行flush或cache之后,该Session中的所有Cache就将清空,默认打开一级缓存。
- 二级缓存是基于namespace和mapper的作用域起作用的,不是依赖于SQL session,默认也是采用PerpetualCache,HashMap存储,需要单独开启,一个是核心配置,一个是mapper映射文件。
18、二级缓存的注意事项
- 对于缓存数据更新机制,当某一个作用域(一级缓存Session/二级缓存Namespaces)的进行了新增、修改、删除操作后,默认该作用域下所有select中的缓存将被clear
- 二级缓存需要缓存的数据实现Serializable接口
- 只有会话提交或者关闭之后,一级缓存中的数据才会转移到二级缓存中。