Spring源码解读——高频面试题

Spring  IoC的底层实现

1.先通过createBeanFactory创建出一个Bean工厂(DefaultListableBeanFactory)

2.开始循环创建对象,因为容器中的bean默认都是单例的,所以优先通过getBean、doGetBean从容器中查找,如果找不到的话;

3.通过createBean、doCreateBean方法,以反射的方式创建对象(一般情况下使用的是无参的构造方法),getDeclaredConstructor、newInstance

4.进行对象的属性填充populateBean(自定义属性赋值)、initializeBean(容器对象的属性赋值)

三级缓存的放置时间和删除时间

三级缓存:createBeanInstance之后(addSingletonFactory方法)

二级缓存:第一次从三级缓存确定对象是代理对象还是普通对象的时候,放入二级缓存同时删除三级缓存(getSingleton方法)

一级缓存:生成完整成品对象之后放入一级缓存,同时删除二级和三级缓存(addSingleton方法)

Spring中用到的设计模式

单例模式:Bean默认都是单例的;

原型模式:指定作用域为prototype;

工厂模式:BeanFactory;

模板方法模式:postProcessBeanFactory,onRefresh,initPropertyValue,等等,体现了Spring的扩展性;

策略模式:XmlBeanDefinitionReader,PropertiesBeanDefinitionReader;

观察者模式:Listener,event,multicast;

适配器模式:Adpater,.....AdviceAdpater;

装饰者模式:BeanWrapper;

责任链模式:使用aop的时候会先生成一个拦截器链

代理模式:动态代理;

委托者模式:delegate;

Spring的事务是如何回滚的

也就是问Spring的事务管理是如何实现的?

我们都知道事务是Aop里面的经常使用的,并且很关键的一个应用场景,但是我们光回答Aop是不够的;

其实整个事务的流程无非就是下面这样

如果是我们自己来实现,准备工作:我们可以使用beforeAdvice来处理,在执行成功之后我们可以使用afterAdvice来处理,如果执行失败了我们可以使用afterThrowing来处理;

但是如果每个事务都这样来做的话就太麻烦了,因为这里面有一堆通知,它必然会构成一个链式结构;

因此我们可以把这些合并到一起去,形成一个完整的结构;

总:spring的事务是由aop来实现的,首先要生成具体的代理对象,然后按照aop的整套流程来执行具体的操作逻辑,正常情况下要通过通知来完成核心功能,但是在aop中事务不是通过通知来实现的,而是通过一个TransactionInterceptor来实现的,它会调用里面的invoke方法来实现具体的逻辑

 

分:

1.先做准备工作,解析各个方法上事务相关的属性,根据具体的属性来判断是否开始新事务;

2.当需要开启的时候,获取数据库连接,关闭自动提交功能,开启事务;

3.执行具体的逻辑操作;

4.在操作过程中,如果执行失败了,那么会通过completeTransactionAfterThrowing方法完成事务的回滚操作,回滚的具体逻辑是通过doRollBack方法来实现的,实现的时候也是要先获取连接对象,通过连接对象来回滚

5.如果执行过程中,没有任何意外情况的发生,那么通过commitTransactionAfterReturning方法来完成事务的提交操作,具体逻辑是通过doCommit方法来实现的,实现的时候也是要先获取连接对象,通过连接对象来提交

6.当事务执行完毕之后需要清楚相关的事务信息,通过cleanupTransactionInfo方法

 

rollback=>processRollBack=>doRollBack;

commit=》processCommit=》doCommit;

 

 

 如果想要聊的更加细致的话,需要知道TransactionInfo,TransactionStatus(比较复杂);

Spring的事务传播

有七种传播特性;

PROPAGATION_REQUIRES:默认的Spring事务传播级别,若当前存在事务,则加入该事务,若不存在事务,则新建一个事务

PROPAGATION_REQUIRE_NEW:若当前没有事务则新建一个事务,若当前存在事务则新建一个事务,新老事务相互独立,外部事务抛出异常回滚不会影响内部事务的正常提交

PROPAGATION_NESTED:若当前存在事务,则嵌套在当前事务中执行,若当前没有事务则新建一个事务,类似于REQUIRE_NEW

PROPAGATION_SUPPORTS:支持当前事务,若当前不存在事务,则以非事务的方式执行

PROPAGATION_NOT_SUPPORTED:以非事务的方式执行,若当前存在事务,则把当前事务挂起

PROPAGATION_MANDATORY:强制事务执行,若当前不存在事务,则抛出异常

PROPAGATION_NEVER:以非事务的方式执行,若当前存在事务则抛出异常

某一个事务嵌套另一个事务的时候怎么办?

A方法调用B方法,AB方法都有事务,并且传播特性不同,如果A/B有异常、B/A怎么办?

总:事务的传播特性指的是不同方法的嵌套调用过程中,事务应该如何进行处理,是用同一个事务还是不同的事务,当出现异常的时候是会回滚还是提交,这两个方法之间的相互影响,在日常工作中,使用比较多的是Required,Requires_new,nested;

分:

1.先说事务的不同分类,可以分为三类:支持当前事务,不支持当前事务,嵌套事务

2.如果外层方法是Required,内层方法是Required,Requires_new,nested

3.如果外层方法是Requires_new,内层方法是Required,Requires_new,nested

4.如果外层方法是nested,内层方法是Required,Requires_new,nested

核心处理逻辑非常简单:

1.判断内外方法是否是同一个事务;

1.1 如果是同一个事务,异常统一在外层方法处理

1.2 如果不是同一个事务,内层方法有可能影响到外层方法,但是外层方法是不会影响内层方法的

(大致可以这么理解,但是有个别情况不同,例如nested

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Strine

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值