spring框架中的单例bean是线程安全的吗
不是线程安全的
Spring框架中有一个@Scope注解,默认的值是singleton,单例的。
因为一般在spring的bean中都是注入的无状态的对象,没有线程安全问题,如果在bean中定义了可修改的成员变量,是要考虑线程安全问题的,可以使用多例或者加锁来解决。
AOP
什么是AOP
AOP称为面向切面编程,用于将那些与业务无关,但却对多个对象产生影响的公共行为和逻辑,抽取并封装为一个可重用的模块,这个模块被命名为“切面”(Aspect) ,减少系统中的重复代码,降低了模块间的耦合度,同时提高了系统的可维护性。
常见的AOP使用场景::
- 记录操作日志
- 缓存处理
- Spring中内置的事务处理
记录操作日志思路
获取请求的用户名,请求方式,访问地址,模块名称,登录ip,操作时间,记录到数据库的日志表中
Spring中的事务是如何实现的
Spring支持编程式事务和声明式事务管理两种方式
编程式事务控制:需使用Transaction Template来进行实现,对业务代码有侵入性,项目中很少使用
声明式事务管理:声明式事务管理建立在AOP之上,其本质是通过AOP功能,对方法前后进行拦截,将事务处理的功能编织到拦截的方法中,也就是在目标方法开始之前加入一个事务,在执行完目标方法之后根据执行情况提交或回滚事务。
事务失效的场景
异常捕获处理
原因:事务通知只有捉到了目标抛出的异常,才能进行后续的回滚处理,如果目标自己处理异常,事务通知无法知悉。
解决:在catch块中添加throw new RuntimeException(e)抛出
抛出检查异常
原因:Spring默认只会回滚非检查异常
解决:配置rollbackFor属性
@Transactional(rollbackFor=Exception.class)
非public方法导致的事务失效
原因:Spring为方法创建代理,添加事务通知,前提条件都是该方法是public的
解决:改为public方法
bean的生命周期
BeanDefinition
Spring容器在进行实例化时,会将xml配置的<bean>
的信息封装成一个BeanDefinition对象,Spring根据BeanDefinition来创建Bean对象,里面有很多的属性用来描述Bean
- beanClassName:bean的类名
- initMethodName:初始化方法名称
- properryValues:bean的属性值
- scope:作用域
- lazyinit:延迟初始化
循环引用
在创建A对象的同时需要使用B对象,在创建B对象的同时需要使用A对象
什么是Spring的循环依赖
三级缓存解决循环依赖
Spring解决循环依赖是通过三级缓存,对应的三级缓存如下所示:
缓存名称 | 源码名称 | 作用 |
---|---|---|
一级缓存 | singletonObjects | 单例池,缓存已经经历了完整的生命周期,已经初始化完成的bean的对象 |
二级缓存 | earlySingletonObjects | 缓存早期的bean对象(生命周期还没走完) |
三级缓存 | singletonFactories | 缓存的是ObejctFactory,表示对象工厂,用来创建某个对象的 |
一级缓存作用:限制bean在benaFactory中只存一份,即实现singletonscope,解决不了循环依赖
如果想要打破循环依赖,就需要一个中间人的参与,这个中间人就是二级缓存。
三级缓存解决循环依赖
构造方法出现了循环依赖怎么解决
解决方法:
SpringMVC的执行流程
Springboot自动配置原理
Spring框架常见注解
Spring的常见注解
注解 | 说明 |
---|---|
@Component,@Controllerm@Service,@Repository | 使用在类上用于实例化bean |
@Autowired | 使用在字段上用于根据类型依赖注入 |
@Qualifer | 结合@Autowired一起使用用于根据名称进行依赖 注入 |
@Scope | 标注bean的范围 |
@Configuration | 标注当前类是一个Spring配置类,当创建容器会从该类上加载注解 |
@ComponentScan | 用于指定Spring在初始化容器时要扫描的包 |
@Bean | 使用在方法上,标注将该方法的返回值存储到Spring容器中 |
@Import | 使用@Import导入的类会被Spring加载到IOC容器中 |
@Aspect,@Before,@After,@Around,@Pointcut | 用于切面编程(AOP) |
SpringMVC常见的注解
注解 | 说明 |
---|---|
@RequestMapping | 用于映射请求路径,可以定义在类上和方法上,用于类上,则表示类中的所有的方法都是以该地址作为父路径 |
@RequestBody | 注解实现接收http请求的json数据,将json转换为java对象 |
@RequestParam | 指定请求参数的名称 |
@PathViriable | 从请求路径下中获取请求参数(/user/{id{),传递给方法的形式参数 |
@ResponseBody | 注解实现将controller方法返回对象转换为json对象响应给客户端 |
@RequstHeader | 获取指定的请求头数据 |
@RestController | @Controller+@ResponseBody |
Springboot常见的注解
注解 | 说明 |
---|---|
@SpringbootConfiguration | 组合了-@Configuration注解,实现配置文件的功能 |
@EnableAutoConfiguration | 打开自动配置的功能,也可以关闭某个自动配置的选项 |
@ComponentScan | Spring组件扫描 |
MyBatis执行流程
MyBatis是否支持延迟加载
MyBatis支持延迟加载,但默认没有开启
查询用户的时候,把用户所属的订单数据也查询出来,这个是立即加载
查询用户的时候,暂时不查询订单数据,当需要订单的时候,再查询订单,这个就是延迟加载
延迟加载的原理:
- 使用CGLIB创建目标对象的代理对象
- 当调用方法user.getOrderList()时,进入拦截器invoke方法,发现user.getOrderList()是null值,执行sql查询order列表
- 把order查询上来,然后调用user.setOrderList(),接着完成user.getOderList方法的调用
MyBatis的一二级缓存
一级缓存:基于PerpetualCache的HashMap本地缓存,其存储作用于为Session,当Session进行flush或close之后,该Session中的所有Cache就将清空,默认打开一级缓存
二级缓存:二级缓存是基于namespace和mapper的作用域起作用的,不是依赖于SQL session,默认也是采用PerpetualCache,HashMap存储
注意:
- 对于缓存数据更新机制,当某一个作用域(一级缓存 Session/二级缓存Namespaces)的进行了新增,修改,删除操作后,默认该作用域下偶有select中的缓存将被clear
- 二级缓存需要缓存的数据实现Serializable接口
- 只有会话提交或关闭后,一级缓存中的数据才会转移到二级缓存中