spring refresh流程 事务失效场景

1. prepareRefresh

创建和准备Environment对象,键值对


2. obtainFreshBeanFactory 

获取或创建BeanFactory

beanfactory的作用是负责bean的创建、依赖注入和初始化

beanFactory作为bean的设计蓝图,规定了bean的特征,如单例多例、依赖关系、初始销毁方法等

beanDefinition的来源有多种多样,可以是通过xml获取、通过配置类获得、通过组件获得,


3.prepareBeanFactory

完善beanFactory

StandardBeanExpressionResolver解析SpEL

ResourceEditorRegistrar会注释类型转换器


4. postProcessBeanFactory

空实现,子类扩展

体现模板方法设计模式


5. invokeBeanFactoryPostProcessors

beanFactory后置处理器,充当beanfactory的扩展点,补充和修改beanDefinition


6. registerBeanPostProcessors

bean后处理器,充当bean的扩展点,可工作在bean的实例化、依赖注入、初始化阶段

AutowiredAnnotationBeanPostProcessor:解析@Autowired, @Value注解

CommonAnnotationBeanPostProcessor:解析@Resource, @PostConstruct,@PreDestroy

AnnotationAwareAspectJAutoProxyCreator:为符合切点的目标bean自动创建代理


7. initMessageSource

国际化


8. initApplicationEventMulticaster

事件广播器 

发布事情给监听器


9. onRefresh

空实现,子类扩展

Springboot中子类准备WebServer, 内嵌web容器


10. registerListeners

事件监听器

接收事件


11. finishBeanFactoryInitialization

初始化单例bean ,执行bean后处理器扩展

内嵌值解析器

单例池 singletonObjects


12. finishRefresh

lifecycleProcessor生命周期处理器

用来控制容器内需要生命周期管理的bean


bean生命周期

按scope创建bean

1.创建singleton(单例)

2.创建prototype(多例)

3.自定义scope


创建bean:

1.创建bean实例 @autowired

2.依赖注入-@autowired @value @resource byName byType 

3.初始化 -Aware接口、@PostConstruct InitalizingBean initMethod

4.登记可销毁bean


类型转换


销毁bean


spring 事务失效场景和原因

1.抛出检查异常导致事务不能正常回滚

原因:spring默认只会回滚非检查异常runtimeExecption、error

解决:@Transactional注解上配置rollbackFor属性


2.业务方法内自己try-catch异常导致事务不能正常回滚

原因:事务通知只有在捉到目标抛出的异常,才能进行后续的回滚处理,如果目标自己处理掉异常,事务通知无法知悉

解决1:异常抛出

解决2:手动设置TransactionStatus.setRollbackOnly()


3.aop切面顺序导致事务不能正常回滚

原因:事务切面优先级最低,但如果自定义切面优先级和他一样,则还是自定义切面在内层,这时若自定义切面没有正常抛出异常。。

解决:同2


4.非public方法导致事务失效

原因:spring为方法创建代理、添加事务通知、前提条件是该方法是public的

解决:改为public方法


5.父子容器导致事务失效

原因:子容器扫描范围过大,把未加事务配置的service扫描进来

解发:各扫描各的,

解法2:不要用父子容器,bean都放在一个容器里


6.调用本类方法导致传播行为失效

原因:本类方法调用不经过代理,因此无法增强

解法1:依赖注入自己(代理)来调用

解法2:通过AopContext拿到代理对象,来调用


7. @Transactional没有保证原子行为

原因:事务原子性仅涵盖insert、update、delete、select ...for update语句,select方法并不阻塞

8.@Transactional方法导致synchronized失效

原因:synchronized保证的仅是目标方法的原子性,环绕目标方法的还有commit等操作,它们并未处于sync块内

解决1:synchronized范围扩大至代理方法调用

解决2:使用select.. for update 代替select

你可能感兴趣的:(spring refresh流程 事务失效场景)