欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

解讀Spring接口方法加@Transactional失效的原因

 更新時(shí)間:2023年03月13日 14:28:10   作者:Galen-gao  
這篇文章主要介紹了Spring接口方法加@Transactional失效的原因解讀,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

問題

今天項(xiàng)目測試一個(gè)方法的時(shí)候,發(fā)現(xiàn)日志報(bào)錯(cuò)

日志報(bào)錯(cuò)大致如下:Connection is read-only. Queries leading to data modification are not allowed

org.springframework.dao.TransientDataAccessResourceException:
### Error updating database.  Cause: java.sql.SQLException: Connection is read-only. Queries leading to data modification are not allowed
### The error may involve com.o2o.app.repository.AccountOrderMybatisDao.updateOrder-Inline
### The error occurred while setting parameters
### SQL: UPDATE t_account_order SET ORDER_STATUS=?, STATUS_DESCRIPTION=?, IS_DELETE = ?           WHERE  TRADE_CODE = ? and TRADE_TYPE=?
### Cause: java.sql.SQLException: Connection is read-only. Queries leading to data modification are not allowed; SQL []; Connection is 
read-only. Queries leading to data modification are not allowed; nested exception is java.sql.SQLException: 
Connectionis read-only. Queries leading to data modification are not allowed

第一眼看上去,這不是Spring事務(wù)配置了只讀事務(wù)屬性,當(dāng)執(zhí)行sql寫操作當(dāng)然會(huì)失敗了,為了更好解決這個(gè)bug,讓我們先溫習(xí)一下事務(wù)以及Spring在事務(wù)傳播做了那些事?

  • 事務(wù)介紹
  • 事務(wù)(Transaction):指的是要做的事情,在計(jì)算機(jī)術(shù)語指的是訪問并能更新數(shù)據(jù)庫數(shù)據(jù)的一個(gè)程序執(zhí)行單元
  • 由于我們?nèi)粘i_發(fā),需要經(jīng)常對(duì)關(guān)系型數(shù)據(jù)庫打交道,這里簡單介紹一下關(guān)系型數(shù)據(jù)庫的事務(wù)四大屬性
事務(wù)名稱解釋
原子性(Atomicity)事務(wù)是一個(gè)原子操作,原子操作簡單理解指的是這個(gè)操作要么全部成功,要么全部失敗
一致性(Consistency)事務(wù)無論成功與否,數(shù)據(jù)庫必須保證所處的數(shù)據(jù)不應(yīng)被破壞,舉個(gè)例子:A給B無論成功或失敗轉(zhuǎn)賬,那么A的錢+B的錢前后應(yīng)該總和相等
隔離性(Isolation)同一份的數(shù)據(jù)可能有很多事務(wù)進(jìn)行操作,因此要將各種事務(wù)隔離開,防止數(shù)據(jù)被損壞
持久性(Durability)事務(wù)如果一旦完成,結(jié)果都應(yīng)不變,因?yàn)檫@樣無論系統(tǒng)發(fā)送了什么錯(cuò)誤,都能進(jìn)行數(shù)據(jù)恢復(fù)
  • Spring事務(wù)核心類和接口
  • 如下圖所示:在spring-tx包下有三個(gè)spring事務(wù)管理非常重要的接口 : PlatformTransactionManager,TransactionDefinition,TransactionStatus


spring并不實(shí)現(xiàn)各個(gè)數(shù)據(jù)庫持久層的事務(wù)實(shí)現(xiàn),而是提供對(duì)應(yīng)的事務(wù)管理器,如下圖所示:


我們首先查看PlatformTransactionManager接口的源代碼:

TransactionStatus getTransaction(TransactionDefinition definition)

官方的解釋是: Return a currently active transaction or create a new one, according to the specified propagation behavior

這句話的意思是根據(jù)指定的事務(wù)行為,返回當(dāng)前的事務(wù)或者新建一個(gè)事務(wù)。

void commit(TransactionStatus status)

官方解釋是:Commit the given transaction, with regard to its status. If the transaction has been marked rollback-only programmatically, perform a rollback.

這句話的意思是根據(jù)事務(wù)的狀態(tài)提交事務(wù),如果事務(wù)標(biāo)記了rollback-only,請(qǐng)執(zhí)行會(huì)滾。

void rollback(TransactionStatus status)

官方的解釋是: Perform a rollback of the given transaction,即對(duì)事務(wù)進(jìn)行回滾。

看到這里,我們需要明確三個(gè)接口中入?yún)⒌腡ransactionDefinition是個(gè)什么東西呢?

讓我們先大致查看一下TransactionDefinition接口的方法和成員變量,以下將會(huì)對(duì)此接口的方法做個(gè)簡單的介紹,如下圖所示:

  • 事務(wù)的傳播行為
  • 當(dāng)事務(wù)方法被調(diào)用時(shí)候,必須指定事務(wù)如何傳播,下面是事務(wù)傳播行為的介紹:
事務(wù)名稱解釋
PROPAGATION_REQUIRED支持當(dāng)前的事務(wù),如果當(dāng)前事務(wù)不存在就新建一個(gè)事務(wù)
PROPAGATION_SUPPORTS支持當(dāng)前事務(wù),如果事務(wù)不存在,將以非事務(wù)方式運(yùn)行
PROPAGATION_MANSATORY支持當(dāng)前事務(wù),如果事務(wù)不存在將拋異常
PROPAGATION_REQUIRES_NEW如果當(dāng)前事務(wù)存在,將當(dāng)前事務(wù)掛起并創(chuàng)建新的事務(wù),如果當(dāng)前事務(wù)不存在就新建一個(gè)事務(wù)
PROPAGATION_NOT_SUPPORTED不支持當(dāng)前事務(wù),以非事務(wù)的方式運(yùn)行
PROPAGATION_NEVER不支持當(dāng)前事務(wù),如果當(dāng)前事務(wù)存在就拋異常
PROPAGATION_NESTED如果當(dāng)前事務(wù)存在,則執(zhí)行一個(gè)內(nèi)嵌的事務(wù)
  • 事務(wù)的隔離級(jí)別
  • 典型的事務(wù)隔離不同所造成問題如下:

1.臟讀:臟讀發(fā)送在A事務(wù)讀取B事務(wù)已經(jīng)改寫但是還未提交的數(shù)據(jù),若此時(shí)B事務(wù)回滾了,那么A事務(wù)獲取就是臟數(shù)據(jù)

2.不可重復(fù)讀:不可重復(fù)讀發(fā)送在當(dāng)A事務(wù)執(zhí)行2次查詢,每一次獲取的數(shù)據(jù)結(jié)果都不相同,這是由于B事務(wù)在A事務(wù)2次查詢期間進(jìn)行了更新

3.幻讀: 幻讀發(fā)送在當(dāng)A事務(wù)讀取了幾行數(shù)據(jù),緊接著B事務(wù)進(jìn)行輸入的插入,在隨后的查詢中A事務(wù)就會(huì)讀了原本不存在的記錄

不可重復(fù)讀特指修改的記錄,而幻讀指的是新增或刪除的記錄

  • 只讀屬性
  • 如果設(shè)置了只讀事務(wù),只讀事務(wù)常常用于做查詢使用,此時(shí)的增刪改,將會(huì)報(bào)Connection is read-only. Queries leading to data modification are not allowed的異常。
  • 事務(wù)的超時(shí)
  • 一個(gè)正常和良好的程序,事務(wù)的行為時(shí)間并不會(huì)很長,較長的事務(wù)運(yùn)行時(shí)間,會(huì)占用數(shù)據(jù)庫資源,所以這里就設(shè)置超時(shí)時(shí)間,若指定時(shí)間內(nèi)沒有執(zhí)行完事務(wù),將會(huì)自動(dòng)進(jìn)行回滾
  • 事務(wù)的名稱
  • 在一個(gè)事務(wù)行為中配置獲取事務(wù)的名稱,如我們常見的save,add,del 等等…

以上溫習(xí)過Spring事務(wù)管理器和傳播行為后,所以既然報(bào)錯(cuò)Connection is read-only. Queries leading to data modification are not allowed所以我們?cè)诮涌诜椒ǖ膶?shí)現(xiàn),加了以下的注解: @Transactional(propagation = Propagation.REQUIRED, readOnly = false),but!當(dāng)我再次請(qǐng)求接口的時(shí)候,發(fā)現(xiàn)依然還是報(bào)同樣的錯(cuò)誤,百度一下,發(fā)現(xiàn)有相關(guān)問題博客的收集:Spring下默認(rèn)事務(wù)機(jī)制中@Transactional 無效的原因

Method visibility and @Transactional When using proxies, you should apply the @Transactional annotation 
only to methods with public visibility. If you do annotate protected, private or package-visible methods with the
@Transactional annotation, no error is raised, but the annotated method does not exhibit the configured transactional settings. 
Consider the use of AspectJ (see below) if you need to annotate non-public methods.

大概意思是:如果你是使用默認(rèn)的Spring Aop代理方式將@Transaction注解應(yīng)該用于公共可見即public的,如果對(duì)protected,或者private的方法加入@Transaction注解,則會(huì)無效。如果想在私有方法是使事務(wù)有效,可以用AspectJ進(jìn)行實(shí)現(xiàn)。

but,我們的注解沒有加在protected和private方法上,但是依然無效,why?

此次注解失效原因像下面簡單的例子一樣:在電商系統(tǒng)中,存在待支付的訂單,假設(shè)有一個(gè)訂單編號(hào)為201904191102的訂單要進(jìn)行支付,首先我需要刷新支付頁面,就需要調(diào)用收銀臺(tái)接口,由于一直使用的scala開發(fā),所以下面的代碼使用scala做演示:

  /**
    * 刷新收銀臺(tái)的接口:refreshCashier
    */
  def refreshCashier(orderId: String): OrderInfo
  
    /**
    * 更新錢包的方法
    */
  def updateWallet(order:orderInfo):PayDto
  
  override def refreshCashier(orderId: String): OrderInfo = {
    // 偽代碼,在這個(gè)方法里面調(diào)用 updateWallet方法
  }
 @Transactional(propagation = Propagation.REQUIRED, readOnly = false)
  override def updateWallet(order:orderInfo): PayDto = {
   // 這里發(fā)送了異常
   throw new AppException()
  }

在上面的代碼中,refreshCashier方法調(diào)用了updateWallet方法的時(shí)候,當(dāng)updateWallet方法出錯(cuò)報(bào)異常,事務(wù)并沒有回滾,這是因?yàn)镾pring Aop動(dòng)態(tài)代理會(huì)為每個(gè)class對(duì)象生成代理對(duì)象,只有在代理對(duì)象之間進(jìn)行調(diào)用的時(shí)候,將會(huì)觸發(fā)切面相關(guān)的邏輯處理。

所以要保證整個(gè)方法調(diào)用鏈的事務(wù)性,在refreshCashier方法加上@Transaction注解,此時(shí)才能保證,updateWallet方法出錯(cuò)時(shí)候,整個(gè)方法能進(jìn)行事務(wù)的回滾。這樣完美,問題解決了.

總結(jié)

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • java的JIT 工作原理簡單介紹

    java的JIT 工作原理簡單介紹

    這篇文章主要介紹了java的JIT 工作原理簡單介紹的相關(guān)資料,需要的朋友可以參考下
    2017-03-03
  • Mybatis-Plus自動(dòng)生成的數(shù)據(jù)庫id過長的解決

    Mybatis-Plus自動(dòng)生成的數(shù)據(jù)庫id過長的解決

    這篇文章主要介紹了Mybatis-Plus自動(dòng)生成的數(shù)據(jù)庫id過長的解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-12-12
  • DoytoQuery 聚合查詢方案示例詳解

    DoytoQuery 聚合查詢方案示例詳解

    這篇文章主要為大家介紹了DoytoQuery 聚合查詢方案示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-12-12
  • 淺談Java中static關(guān)鍵字的作用

    淺談Java中static關(guān)鍵字的作用

    這篇文章主要介紹了Java中static關(guān)鍵字的作用,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-03-03
  • Java實(shí)現(xiàn)二叉堆、大頂堆和小頂堆

    Java實(shí)現(xiàn)二叉堆、大頂堆和小頂堆

    二叉堆就是完全二叉樹,或者是靠近完全二叉樹結(jié)構(gòu)的二叉樹。大頂堆要求對(duì)于一個(gè)節(jié)點(diǎn)來說,它的左右節(jié)點(diǎn)都比它??;小頂堆要求對(duì)于一個(gè)節(jié)點(diǎn)來說,它的左右節(jié)點(diǎn)都比它大。本文將用Java分別實(shí)現(xiàn)二叉堆、大頂堆和小頂堆。需要的可以參考一下
    2022-01-01
  • Java?中泛型?T?和???的區(qū)別詳解

    Java?中泛型?T?和???的區(qū)別詳解

    本文主要介紹了Java?中泛型?T?和???的區(qū)別,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-01-01
  • MyBatis查詢結(jié)果resultType返回值類型的說明

    MyBatis查詢結(jié)果resultType返回值類型的說明

    這篇文章主要介紹了MyBatis查詢結(jié)果resultType返回值類型的說明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2020-11-11
  • 淺談java 重寫equals方法的種種坑

    淺談java 重寫equals方法的種種坑

    這篇文章主要介紹了淺談java 重寫equals方法的種種“坑”,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-01-01
  • javaweb 實(shí)現(xiàn)文件下載的方法及實(shí)例代碼

    javaweb 實(shí)現(xiàn)文件下載的方法及實(shí)例代碼

    這篇文章主要介紹了javaweb 實(shí)現(xiàn)文件下載的方法的相關(guān)資料,這里提供了實(shí)現(xiàn)代碼,需要的朋友可以參考下
    2016-11-11
  • SpringBoot整合MyBatis Plus實(shí)現(xiàn)基本CRUD與高級(jí)功能

    SpringBoot整合MyBatis Plus實(shí)現(xiàn)基本CRUD與高級(jí)功能

    Spring Boot是一款用于快速構(gòu)建Spring應(yīng)用程序的框架,而MyBatis Plus是MyBatis的增強(qiáng)工具,本文將詳細(xì)介紹如何在Spring Boot項(xiàng)目中整合MyBatis Plus,并展示其基本CRUD功能以及高級(jí)功能的實(shí)現(xiàn)方式,需要的朋友可以參考下
    2024-02-02

最新評(píng)論