- M2方法的預期作用是根據id值(主鍵)加行鎖,整個方法放在事務中執行
- 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); // 其它的一些事務操作 } }
select * from t1 where id=1 for update --readonly
- 數據庫是一主多從結構,讀寫分離
- 方法上加@Transactional註解,不管是查詢還是增刪改,都可以保證走主庫,可是異常提示"readonly",說明select語句走了從庫
- M1和M2在同一個類C中,this.M2()語句並沒有使用類C在容器中的代理對象,而是使用原生對象調用M2方法,故@Transactional註解未生效
- 由於@Transactional未生效,又因爲"select * from t1 where id=1 for update"是讀語句,會主動走從庫,從庫又是隻讀的,所以會拋出readonly異常
- 單獨的類中調用:在單獨的類中使用類C的代理對象調用M2方法
- 類C中操作:從IOC容器獲取代理對象
public void M1() { C c = SpringUtils.getBean(C.class); c.M2(); }
- 類C中操作:在類中定義代理對象並使用
@Autowired private C c1; public void M1() { c1.M2(); }