一、Spring框架中的单例bean的安全性
Spring框架中有一个@Scope注解,默认的值就是singleton,单例的;因为一般在spring的bean中注入的都是无状态的对象,所以没有线程安全问题。但是如果在bean中定义了可修改的成员变量,是需要考虑线程安全问题的,可以使用多例或者加锁解决
二、AOP
AOP称为面向切面编程,用于将那些与业务无关,但却对多个对象产生影响的公共行为和逻辑,抽取并封装为一个可重用的模块,这个模块被命名为"切面"(Aspect),可以减少系统中的重复代码。降低模块间的耦合度,同时提高系统的可维护性
常见的AOP使用场景:记录操作日志、缓存处理、Spring中内置的事务处理
核心是:使用aop中的环绕通知和切点表达式,通过环绕通知的参数获取请求方法的参数(类、方法、注解、请求方式等),获取到这些参数后保存到数据库
Spring中的事务是如何实现的:
Spring支持编程式事务管理和声明式事务两种方式
编程式事务控制:需使用TransactionTemplate来进行实现,对业务代码有侵入性,项目中很少使用
声明式事务管理:声明式事务管理建立在AOP之上的。其本质是通过AOP功能,对方法前后进行拦截,将事务处理的功能编织到拦截的方法中,也就是在目标方法开始之前加入一个事务,在执行完目标方法之后根据执行情况提交或者回滚事务。
三、Spring中事务失效的场景:
1.异常捕获处理:
事务通知只有捉到了目标抛出的异常,才能进行后续的回滚处理,如果目标自己处理掉异常,事务通知无法知悉;可以在catch中添加throw new RuntimeException(e)进行抛出
2.抛出检查异常:
Spring只会回滚非检查异常,可以通过配置rollbackFor属性解决:
@Transactional(rollbackFor = Exception.class)
3.非public方法导致的事务失效:
Spring为方法创建代理、添加事务通知、前提条件都是该方法是public的,可以将该方法改成public解决
四、Sping中bean的生命周期:
BeanDefinition:Spring容器在进行实例化时,会将xml配置的<bean>信息封装成一个BeanDefiniion对象,Spring根据BeanDefinition来创建Bean对象,里面有很多的属性用来描述Bean
beanClassName:bean的类名
initMethodName:初始化方法名称
propertyValues:bean的属性值
scope:作用域
lazyInit:延迟初始化
(1).通过BeanDefinition获取bean的定义信息
(2).调用构造函数实例化bean
(3).bean的依赖注入
(4).处理Aware接口(BeanNameAware,BeanFactoryAware,ApplicationContextAware)
(5).Bean的后置处理器BeanPostProcessor-前置
(6).初始化方法(InitializingBean、init-method)
(7).Bean的后置处理器BeanPostProcessor-后置
(8).销毁bean
五、Spring中的循环引用:
循环依赖其实就是循环引用,也就是两个或两个以上的bean互相持有对方,最终形成闭环,比如A依赖于B,B依赖于A
循环依赖在spring中是允许存在的,spring框架依据三级缓存已经解决了大部分的循环依赖
在创建A对象时需要使用B对象,在创建B对象同时需要使用A对象
Spring解决循环依赖是通过三级缓存:
缓存名称 | 源码名称 | 作用 |
一级缓存 | singletonObjects | 单例池,缓存已经经历了完整的生命周期,已经初始化完成的bean对象 |
二级缓存 | earlySingletonObjects | 缓存早期的bean对象(生命周期还没有走完) |
三级缓存 | singletonFactories | 缓存的是ObjectFactory,表示对象工厂,用来创建某个对象的 |
一级缓存作用:限制bean在beanFactory中只存在一份,即实现singleton scope,解决不了循环依赖
如果要想打破循环依赖,就需要一个中间人的参与,这个中间人就是二级缓存
构造方法出现了循环依赖:
A依赖于B,B依赖于A,注入的方法是构造函数,由于bean的生命周期中构造函数是第一个执行的,spring框架并不能解决构造函数的依赖注入
可以使用@Lazy进行懒加载,什么时候需要对象再进行bean对象的创建
六、SpringMVC的执行流程:
1.视图阶段(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传给ViewResolver(视图解析器)
(9).ViewResolver解析后返回具体的View(视图)
(10).DispatcherServlet根据View进行渲染视图(将模型数据填充到视图中)
(11).DispatcherServlet响应用户
2.前后端分离阶段(接口开发、异步请求):
(1).用户发送出请求到前端控制器DispatcherServlet
(2).DispatcherServlet收到请求调用HandlerMapping(处理器映射器)
(3).HandlerMapping找到具体的处理器,生成处理器对象以及处理器拦截器,再一起返回个DispatcherServlet
(4).DispatcherServlet调用HandlerAdapter(处理器适配器)
(5).HandlerAdapter经过适配调用具体的处理器(Handler/Controller)
(6).方法上添加了@ResponseBody
(7).通过HttpMessageConverter来返回结果转换为JSON并相应
七、Springboot自动配置原理:
1.在Spring Boot项目的引导类上有一个注解@SpringBootApplication,这个注解是对三个注解进行了封装,分别是:@SpringBootConfiguration,@EnableAutoConfiguration和@ComponentScan
2.其中@EnableAutoConfiguration是实现自动化配置的核心注解,该注解通过@Import注解导入对应的配置选择器,内部就是读取了该项目和该项目引用的Jar包的classpath路径下META-INF/spring.factories文件中的所配置的类的全类名,在这些配置类中所定义的Bean会根据条件注解所指定的条件来决定是否需要将其导入到Spring容器中
3.条件判断会有像@ConditionalOnClass这样的注解,判断是否有对应的class文件,如果有则加载该类,把这个配置类的所有的Bean放入spring容器中使用
八、Spring框架常见的注解:
1.Spring常见的注解:
注解 | 说明 |
@Component,@Controller,@Service,@Repository | 使用在类上用于实例化Bean |
@Autowired | 使用在字段上用于根据类型依赖注入 |
@Qualifier | 结合@Autowired一起使用用于根据名称进行依赖注入 |
@Scope | 标注Bean的作用范围 |
@Configuration | 指定当前类是一个Spring配置类,当创建容器时会从该类上加载注解 |
@ComponentScan | 用于指定Spring在初始化容器时要扫描的包 |
@Bean | 使用在方法上,标注该方法的返回值存储到Spring容器中 |
@Import | 使用@Import导入的类会被Spring加载到IOC容器中 |
@Aspect,@Before,@After,@Around,@Pointcut | 用于切面编程 |
2.SpringMVC常见的注解:
注解 | 说明 |
@RequestMapping | 用于请求路径,可以定义在类上和方法上,用于类上则表示类中的所有的方法都是以该地址作为父路径 |
@RequestBody | 注解实现接收http请求的json数据,将json转换为java对象 |
@RequestParam | 指定请求参数的名称 |
@PathVairable | 从请求路径下获取请求参数,传递给方法的形式参数 |
@ResponseBody | 注解实现将controller方法返回对象转换为json对象响应给客户端 |
@RequestHeader | 获取指定的请求头数据 |
@RestController | @Controller+@ResponseBody |
3.SpringBoot常见注解:
注解 | 说明 |
@SpringBootConfiguration | 组合了@Configuration注解,实现配置文件的功能 |
@EnableAutoConfiguration | 打开自动配置的功能,也可以关闭某个自动配置的选项 |
@ComponentScan | Spring组件扫描 |
九、MyBatis执行流程:
1.读取MyBatis配置文件:mybatis-config.xml加载运行环境和映射文件
2.构建会话工厂SqlSessionFactory
3.会话工厂创建SqlSession对象(包含了执行SQL语句的所有方法)
4.操作数据库的接口,Executor执行器,同时负责查询缓存的维护
5.Executor接口的执行方法中有一个MappedStatement类型的参数,封装了映射信息
6.输入参数映射
7.输出结果映射
十、MyBatis是否支持延迟加载:
MyBatis支持延迟加载,所谓的延迟加载就是在需要数据的时候进行加载,不需要用到数据时就不加载数据,MyBatis支持一对一关联对象和一对多关联集合对象的延迟加载;在MyBatis配置文件中,可以配置是否启动延迟加载lazyLoadingEnabled=true|false,默认是关闭的
底层原理:
1.使用CGLIB创建目标对象的代理对象
2.当调用目标方法时,进入拦截器invoke方法,判断是否为空,发现目标方法为null值时,执行sql查询对应的列表
3.获取数据以后,调用set方法设置属性值,再继续查询目标方法,就能够获取到对应的值了
十一、MyBatis的一二级缓存:
本地缓存:基于PerpetualCache,本质是一个HashMap
一级缓存:基于PerpetualCache,本质是一个HashMap,作用域是session级别当session进行flush或close之后,该session中所有的cache就被清空,默认打开一级缓存
二级缓存:作用域是namespace和mapper的作用域,不依赖于session,基于PerpetualCache,本质是一个HashMap,二级缓存默认是关闭的,需要手动开启(核心配置文件+mapper映射文件)
注意事项:
(1).对于缓存数据更新机制,当某一个作用域(一级缓存Session/二级缓存Namespaces)的进行了新增、修改、删除操作后,默认该作用域下所有的select中的缓存将被clear
(2).二级缓存需要缓存的数据实现Serializable接口
(3).只有会话提交或者关闭后,一级缓存中的数据才会被转移到二级缓存中