事务回滚失败的场景和原因

Java开发中遇到事务回滚失败,可以考虑从以下列表中寻找原因:

  1. 数据库引擎是否支持事务(Mysql的MyIsam引擎就不支持事务)

  2. 注解所在的类是否注入spring容器中

  3. 注解所在方法是否为public修饰或者final(方法实现了接口是否为public,方法没有实现接口是否为final) 分析:这是由 Spring AOP 的本质决定的。如果你在 protected、private 或者默认可见性的方法上使用
    @Transactional 注解,这将被忽略,也不会抛出任何异常。

    @Transactional的实现是基于动态代理的,private和final修饰的方法,不会被代理。
    个人观点:在spring中动态代理分为jdk动态代理和cglib动态代理,jdk动态代理要求必须实现接口(所以方法必须是public的),但是cglib动态代理底层则是通过字节码生成被代理类的子类来实现的,这里要求被代理类必须能被继承(所以方法不能被final修饰–final可以限制类被继承)并且方法也只能是public,其他的权限修饰符在继承时访问权限会发生变化,例如protected的,在父类中为public,在子类中就为protected了。

  4. 所用数据源是否加载了事务管理器

  5. 是否发生了方法的自调用(同一个类中的A方法调用B方法) 分析:若同一类中的其他没有@Transactional 注解的方法内部调用有@Transactional 注解的方法,有@Transactional 注解的方法的事务被忽略,不会发生回滚。

  6. 当方法发生异常时,使用try catch捕获了异常,并且catch中没有抛出异常或者手动回滚。 分析:事务的回滚是方法发生异常,在aop的异常通知中进行拦截,回滚,如果方法中捕获了异常,是不会被aop的异常通知拦截到的。如果使用了try
    catch捕获异常,需要在catch中抛出一个异常或者在catch中通过TransactionAspectSupport.currentTransactionStatus().setRollbackOnly()
    设置手动回滚。

在PostgreSQL(简称PG或pgsql)中,事务是数据库操作的一个逻辑单元,用于保证数据的一致性和完整性。如果事务中的某条或多条SQL语句执行失败或者用户决定放弃这些更改,可以使用回滚机制撤销这些操作,回到事务开始前的状态。以下是关于事务回滚的基本概念: 1. **事务开始**:使用`BEGIN`语句开始一个事务,这时数据库将事务状态设置为未提交(uncommitted)。 2. **事务提交**:当所有操作都成功完成,使用`COMMIT`语句正式提交事务,所做更改被保存到数据库中,事务状态变为已提交(committed)。 3. **事务回滚**:如果在事务过程中遇到错误或决定不执行后续操作,使用`ROLLBACK`语句撤销自上一次`BEGIN`以来的所有更改,将事务状态恢复到开始前,就像这些操作从未执行过一样。 4. **自动回滚**:如果在一个事务中发生错误(如违反完整性约束),PostgreSQL也自动回滚事务,而无需显式调用`ROLLBACK`。 5. **保存点**:如果事务中存在多个操作,还可以使用`SAVEPOINT`语句创建一个保存点,然后在需要的地方使用`ROLLBACK TO SAVEPOINT`回滚到特定的保存点。 相关问题: 1. PostgreSQL支持哪些类型的回滚策略? 2. 如何在事务中设置隔离级别以影响回滚行为? 3. 事务中的`SAVEPOINT`有什么作用和使用场景? 4. 当一个事务需要回滚,系统如何保证数据的一致性?
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值