建设通相似的网站,搭建什么平台如何表达,shopwt多用户商城系统,网络运营外包托管1. 数据库引擎不支持事务
这里以 MySQL为例#xff0c;MyISAM引擎是不支持事务操作的#xff0c;一般要支持事务都会使用InnoDB引擎#xff0c;根据MySQL 的官方文档说明#xff0c;从MySQL 5.5.5 开始的默认存储引擎是 InnoDB#xff0c;之前默认的都是 MyISAM#xff…1. 数据库引擎不支持事务
这里以 MySQL为例MyISAM引擎是不支持事务操作的一般要支持事务都会使用InnoDB引擎根据MySQL 的官方文档说明从MySQL 5.5.5 开始的默认存储引擎是 InnoDB之前默认的都是 MyISAM所以这一点要值得注意如果底层引擎不支持事务那么再怎么设置也没有用。
2.没有被 Spring 管理
示例如下:
public class OrderServiceImpl implements OrderService{Transactionalpublic void updateOrder(Order order){//update order}
}如果此时把Service注解注释掉那么这个类就不会被加载成一个Bean这个类就不会Spring管理了事务自然就失效了。
3. 方法不是 public 的
Transactional注解只能用干public 的方法上否则事多不会生效如果要用在非public的方法上则可以开启基于 AspcetJ 框架的静态代理模式。
4.发生自身调用 示例如下:
Service
public class OrderServiceImpl implements OrderService {public void update(Order order) {updateOrder(order);}
}Transactional
public void updateOrder(0rder order) {// update order}
}update 方法上面没有加 Transactional 注解如果调用有 Transactional 注解的updateOrder 方法那么 updateOrder 方法上的事务还可以生效吗? 这里大家可以先想一想后面会揭晓答案。
再来看下面这个例子:
Service
public class OrderServiceImpl implements OrderService {Transactionalpublic void update(Order order) {updateOrder(order);}
}Transactional(propagation Propagation.REQUIRES_NEW)
public void updateOrder(0rder order) {updateOrder(order);}
}这次在 update 方法上加了 Transactional, 如果在 updateOrder 上加了 REOUIRES_NEW新开启一个事务那么新开启的事务可以生效吗?
这两个例子中的事务都不会生效因为它们发生了自身调用就调用了该类自己的方法而没有经过Spring的代理类默认只有调用外部代理类的方法事务才会生效这也是老生常谈的问题了。
这个问题的解决方案之一就是在事务所在的类中注入自己用注入的对象再调用另外一个方法这个不太优雅在Spring 中可以在当前线程中暴露并获取当前代理类通过在启动类上添加以下注解来启用暴露代理类如下面的示例所示。
EnableAspectJAutoProxy(exposeProxy true)然后通过以下代码获取当前代理类并调用代理类的事务方法
((0rderService) AopContext.currentProxy()).updateOrder();Spring 默认只有调用 Spring代理类的public 方法事务才能生效。
异步调用也会没有事务 对于一个事务开子线程
1.为什么会在一个事务想到开子线程
因为在某些场景下信息提交后在保存的过程中部分信息需要大量时间去写业务逻辑然后获取然而这部分信息并不要求实时性所以就想到在一个service下开启子线程处理这块
2.引发问题一个事务下开启子线程处理业务是否会开启新的事务或者共用一个事务
答案都是否定的。
3.验证流程
验证一在springdata下测试
由于springdata在update,delete,save操作时强制要求开启事务所以子线程并没有开启事务 等待足够时间却查不到主线程的未提交事务所以没有公用一个事务
子线程强开事务解决方案可以新建一个service在当前service调用新sevice事务方法就可以开启新事务 5.没有配置事务管理器
如果没有配置以下DataSourceTransactionManager数据源事务管理器那么事务也不会生效
Bean
public PlatformTransactionManager transactionManager(DataSource dataSource) {return new DataSourceTransactionManager(dataSource);
} 但在 Spring Boot 中只要引入了 spring-boot-starter-data-jdbc 启动器依赖就会自动配置DataSourceTransactionManager数据源事务管理器所以 Spring Boot框架不存在这个问题但在传统的 Spring 框架中需要注意。
6. 设置了不支持事务
示例如下:
Service
public class OrderServiceImpl implements OrderService {Transactionalpublic void update(Order order) {updateOrder(order);}Transactional(propagation Propagation.NOT_SUPPORTED)public void updateOrder(Order order) {//update order}
}这里的Propagation.NOT_SUPPORTED表示当前方法不以事务方式运行当前若存在事务则挂起这就是主动不支持以事务方式运行了。
7. 异常没有被抛出
示例如下:
Service
public class OrderServiceImpl implements OrderService {Transactionalpublic void update(Order order) {try{// update order}catch{}}
}这个方法把异常给捕获了但没有抛出来所以事务不会回滚只有捕捉到异常事务才会生效。
8. 异常类型不匹配
示例如下
Service
public class OrderServiceImpl implements OrderService {Transactionalpublic void update(Order order) {try{// update order}catch{throw new Exception(更新失败);}}
}因为 Spring 默认回滚的是 RuntimeException 异常和程序抛出的 Exception 异常不匹配所以事务也是不生效的。如果要触发默认 RuntimeException之外异常的回滚则需要在 Transactiona事务注解上指定异常类示例如下:
Transactional(rollbackFor Exception.class)在今天的文章中总结了使用 Transactional注解导致事务失效的几个常见场景如果 Transactional事务不生效则可以根据这几种情形排查一下其实次数最多的也就是发生自身调用、异常被捕获、异常抛出类型不匹配这几种场景。