【Java】select for upate 引起的问题

本文讨论了在M2方法中使用事务时,由于代码直接调用而非通过事务代理导致事务未生效的问题。作者提供了在类C中通过Spring容器获取代理对象调用M2的方法,以确保事务在数据库读写分离场景下正常工作,避免了readonly异常。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一. 代码说明

  1. M2方法的预期作用是根据id值(主键)加行锁,整个方法放在事务中执行
  2. M1方法调用M2方法
public class C
{
    public void M1()
    {
        this.M2();
    }
    
    @Transactional
    public void M2()
    {
        String sql="select * from t1 where id=1 for update";
        executeSql(sql);
        
        // 其它的一些事务操作
    }
}

二. 问题

1. M2方法事务未生效,并排除异常

select * from t1 where id=1 for update --readonly

三. 原因分析

  1. 数据库是一主多从结构,读写分离
  2. 方法上加@Transactional注解,不管是查询还是增删改,都可以保证走主库,可是异常提示"readonly",说明select语句走了从库
  3. M1和M2在同一个类C中,this.M2()语句并没有使用类C在容器中的代理对象,而是使用原生对象调用M2方法,故@Transactional注解未生效
  4. 由于@Transactional未生效,又因为"select * from t1 where id=1 for update"是读语句,会主动走从库,从库又是只读的,所以会抛出readonly异常

四. 解决方案

  1. 单独的类中调用:在单独的类中使用类C的代理对象调用M2方法
  2. 类C中操作:从IOC容器获取代理对象
public void M1() {
    C c = SpringUtils.getBean(C.class);
    c.M2();
}

     3. 类C中操作:在类中定义代理对象并使用

@Autowired
private C c1;

public void M1() {

    c1.M2();
}

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值