中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

怎么在Spring中利用@Transactional實現事務回滾

發布時間:2021-04-08 17:10:00 來源:億速云 閱讀:1353 作者:Leah 欄目:編程語言

今天就跟大家聊聊有關怎么在Spring中利用@Transactional實現事務回滾,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結了以下內容,希望大家根據這篇文章可以有所收獲。

一、使用場景舉例

在了解@Transactional怎么用之前我們必須要先知道@Transactional有什么用。下面舉個栗子:比如一個部門里面有很多成員,這兩者分別保存在部門表和成員表里面,在刪除某個部門的時候,假設我們默認刪除對應的成員。但是在執行的時候可能會出現這種情況,我們先刪除部門,再刪除成員,但是部門刪除成功了,刪除成員的時候出異常了。這時候我們希望如果成員刪除失敗了,之前刪除的部門也取消刪除。這種場景就可以使用@Transactional事物回滾。

二、checked異常和unchecked異常

這里之所以讓大家清楚checked異常和unchecked異常概念,是因為:

Spring使用聲明式事務處理,默認情況下,如果被注解的數據庫操作方法中發生了unchecked異常,所有的數據庫操作將rollback;如果發生的異常是checked異常,默認情況下數據庫操作還是會提交的。

checked異常:

表示無效,不是程序中可以預測的。比如無效的用戶輸入,文件不存在,網絡或者數據庫鏈接錯誤。這些都是外在的原因,都不是程序內部可以控制的。

必須在代碼中顯式地處理。比如try-catch塊處理,或者給所在的方法加上throws說明,將異常拋到調用棧的上一層。

繼承自java.lang.Exception(java.lang.RuntimeException除外)。

unchecked異常:

表示錯誤,程序的邏輯錯誤。是RuntimeException的子類,比如IllegalArgumentException, NullPointerException和IllegalStateException。

不需要在代碼中顯式地捕獲unchecked異常做處理。

繼承自java.lang.RuntimeException(而java.lang.RuntimeException繼承自java.lang.Exception)。

看下面的異常結構圖或許層次感更加深些:

怎么在Spring中利用@Transactional實現事務回滾

三、@Transactional的使用實例

本實例采用的是eclipse+maven,maven只是作為jar管理,即便不了解的maven的猿友也可以讀懂。

3.1、spring的配置文件

里面必須先配置tx名字空間如下:

怎么在Spring中利用@Transactional實現事務回滾

為了使用基于@Transactional的事務管理,需要在Spring中進行如下的配置:

<bean id="appTransactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
   <property name="dataSource" ref="dataSource" />
  </bean>

  <tx:annotation-driven proxy-target-class="false" transaction-manager="appTransactionManager" />

博主的整個spring配置文件:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"
  xmlns:tx="http://www.springframework.org/schema/tx"
  xmlns:aop="http://www.springframework.org/schema/aop"
  xsi:schemaLocation=" 
      http://www.springframework.org/schema/tx
      http://www.springframework.org/schema/tx/spring-tx-2.5.xsd
      http://www.springframework.org/schema/beans 
      http://www.springframework.org/schema/beans/spring-beans-3.0.xsd 
      http://www.springframework.org/schema/aop 
      http://www.springframework.org/schema/aop/spring-aop-3.0.xsd
      http://www.springframework.org/schema/context 
      http://www.springframework.org/schema/context/spring-context-3.0.xsd">

   <!-- 引入jdbc配置文件 --> 
   <bean id="propertyConfigurer" class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
    <property name="locations">
      <list>
        <value>classpath:properties/*.properties</value>
        <!--要是有多個配置文件,只需在這里繼續添加即可 -->
      </list>
    </property>
  </bean>

  <!-- 配置數據源 -->
  <bean id="dataSource"
    class="org.springframework.jdbc.datasource.DriverManagerDataSource">
    <!-- 不使用properties來配置 -->
    <!-- <property name="driverClassName" value="com.mysql.jdbc.Driver" /> 
      <property name="url" value="jdbc:mysql://localhost:3306/learning" /> 
      <property name="username" value="root" /> 
      <property name="password" value="christmas258@" /> -->
    <!-- 使用properties來配置 -->
    <property name="driverClassName">
      <value>${jdbc_driverClassName}</value>
    </property>
    <property name="url">
      <value>${jdbc_url}</value>
    </property>
    <property name="username">
      <value>${jdbc_username}</value>
    </property>
    <property name="password">
      <value>${jdbc_password}</value>
    </property>
  </bean>

  <bean id="appTransactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
   <property name="dataSource" ref="dataSource" />
  </bean>

  <tx:annotation-driven proxy-target-class="false" transaction-manager="appTransactionManager" />
  <!-- 自動掃描了所有的XxxxMapper.xml對應的mapper接口文件,這樣就不用一個一個手動配置Mpper的映射了,只要Mapper接口類和Mapper映射文件對應起來就可以了。 -->
  <bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
    <property name="basePackage"
      value="com.luo.dao" />
  </bean>

  <!-- 配置Mybatis的文件 ,mapperLocations配置**Mapper.xml文件位置,configLocation配置mybatis-config文件位置-->
  <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
    <property name="dataSource" ref="dataSource" />
    <property name="mapperLocations" value="classpath:mapper/*.xml"/> 
    <property name="configLocation" value="classpath:mybatis/mybatis-config.xml" />
    <!-- <property name="typeAliasesPackage" value="com.tiantian.ckeditor.model" 
      /> -->
  </bean>

  <!-- 自動掃描注解的bean -->
  <context:component-scan base-package="com.luo.service" />
</beans>

3.2、使用@Transactional,在添加用戶實現類方法加上注解

@Transactional(propagation=Propagation.REQUIRED)
public void addUser(User user) {
  userDao.addUser(user);
  String string = null;
  if(string.equals("")) {
    int i = 0;
  }
}

上面的方法我故意讓其出現空指針異常,會事物回滾

3.3、運行單元測試類

@Test 
public void addUserTest(){ 
  User user = new User();
  user.setUserName("luoguohui1");
  user.setUserPassword("luoguohui1");
  userService.addUser(user);
}

發現無法插入進去,但是如果把@Transactional去掉,即代碼如下,雖然出現異常,但是數據庫中還是有添加對應數據的:

怎么在Spring中利用@Transactional實現事務回滾

3.4、源碼下載

本文最終源碼下載:first_maven_project_jb51.rar

四、Spring中的@Transactional必須要了解的概念

Spring中的@Transactional基于動態代理的機制,提供了一種透明的事務管理機制,方便快捷解決在開發中碰到的問題。

一般使用是通過如下代碼對方法或接口或類注釋:

@Transactional(propagation=Propagation.NOT_SUPPORTED)

Propagation支持7種不同的傳播機制:

REQUIRED:如果存在一個事務,則支持當前事務。如果沒有事務則開啟一個新的事務。

SUPPORTS: 如果存在一個事務,支持當前事務。如果沒有事務,則非事務的執行。但是對于事務同步的事務管理器,PROPAGATION_SUPPORTS與不使用事務有少許不同。

NOT_SUPPORTED:總是非事務地執行,并掛起任何存在的事務。

REQUIRESNEW:總是開啟一個新的事務。如果一個事務已經存在,則將這個存在的事務掛起。

MANDATORY:如果已經存在一個事務,支持當前事務。如果沒有一個活動的事務,則拋出異常。

NEVER:總是非事務地執行,如果存在一個活動事務,則拋出異常

NESTED:如果一個活動的事務存在,則運行在一個嵌套的事務中。如果沒有活動事務,則按REQUIRED屬性執行。

下面是一些需要注意的事項,必須必須必須要看,不然遇到各種坑別說博主沒有提醒你哦:

下面是一些需要注意的事項,必須必須必須要看,不然遇到各種坑別說博主沒有提醒你哦:

下面是一些需要注意的事項,必須必須必須要看,不然遇到各種坑別說博主沒有提醒你哦:

在需要事務管理的地方加@Transactional 注解。@Transactional 注解可以被應用于接口定義和接口方法、類定義和類的 public 方法上。

@Transactional 注解只能應用到 public 可見度的方法上。 如果你在 protected、private 或者 package-visible 的方法上使用 @Transactional 注解,它也不會報錯, 但是這個被注解的方法將不會展示已配置的事務設置。

注意僅僅 @Transactional 注解的出現不足于開啟事務行為,它僅僅 是一種元數據。必須在配置文件中使用配置元素,才真正開啟了事務行為。

通過 元素的 “proxy-target-class” 屬性值來控制是基于接口的還是基于類的代理被創建。如果 “proxy-target-class” 屬值被設置為 “true”,那么基于類的代理將起作用(這時需要CGLIB庫cglib.jar在CLASSPATH中)。如果 “proxy-target-class” 屬值被設置為 “false” 或者這個屬性被省略,那么標準的JDK基于接口的代理將起作用。

Spring團隊建議在具體的類(或類的方法)上使用 @Transactional 注解,而不要使用在類所要實現的任何接口上。在接口上使用 @Transactional 注解,只能當你設置了基于接口的代理時它才生效。因為注解是 不能繼承 的,這就意味著如果正在使用基于類的代理時,那么事務的設置將不能被基于類的代理所識別,而且對象也將不會被事務代理所包裝。

@Transactional 的事務開啟 ,或者是基于接口的 或者是基于類的代理被創建。所以在同一個類中一個方法調用另一個方法有事務的方法,事務是不會起作用的。

看完上述內容,你們對怎么在Spring中利用@Transactional實現事務回滾有進一步的了解嗎?如果還想了解更多知識或者相關內容,請關注億速云行業資訊頻道,感謝大家的支持。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

随州市| 边坝县| 柳江县| 准格尔旗| 根河市| 阿拉善盟| 专栏| 灵川县| 建昌县| 蓬安县| 吉水县| 贵南县| 永新县| 精河县| 富裕县| 谷城县| 雷山县| 澜沧| 邢台县| 厦门市| 连江县| 宁津县| 襄城县| 尉氏县| 甘孜县| 云龙县| 普定县| 元朗区| 梁平县| 枣阳市| 互助| 万全县| 甘洛县| 凤台县| 山阴县| 平安县| 阆中市| 扎鲁特旗| 平果县| 玉龙| 定远县|