【問題排查】select for upate 引起的問題

 
 
一. 代碼說明
  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();
}

 

  1. 類C中操作:在類中定義代理對象並使用
@Autowired
private C c1;

public void M1() {

    c1.M2();
}

 

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章