淺談Spring中@Transactional事務(wù)回滾及示例(附源碼)
一、使用場(chǎng)景舉例
在了解@Transactional怎么用之前我們必須要先知道@Transactional有什么用。下面舉個(gè)栗子:比如一個(gè)部門里面有很多成員,這兩者分別保存在部門表和成員表里面,在刪除某個(gè)部門的時(shí)候,假設(shè)我們默認(rèn)刪除對(duì)應(yīng)的成員。但是在執(zhí)行的時(shí)候可能會(huì)出現(xiàn)這種情況,我們先刪除部門,再刪除成員,但是部門刪除成功了,刪除成員的時(shí)候出異常了。這時(shí)候我們希望如果成員刪除失敗了,之前刪除的部門也取消刪除。這種場(chǎng)景就可以使用@Transactional事物回滾。
二、checked異常和unchecked異常
這里之所以讓大家清楚checked異常和unchecked異常概念,是因?yàn)椋?
Spring使用聲明式事務(wù)處理,默認(rèn)情況下,如果被注解的數(shù)據(jù)庫操作方法中發(fā)生了unchecked異常,所有的數(shù)據(jù)庫操作將rollback;如果發(fā)生的異常是checked異常,默認(rèn)情況下數(shù)據(jù)庫操作還是會(huì)提交的。
checked異常:
表示無效,不是程序中可以預(yù)測(cè)的。比如無效的用戶輸入,文件不存在,網(wǎng)絡(luò)或者數(shù)據(jù)庫鏈接錯(cuò)誤。這些都是外在的原因,都不是程序內(nèi)部可以控制的。
必須在代碼中顯式地處理。比如try-catch塊處理,或者給所在的方法加上throws說明,將異常拋到調(diào)用棧的上一層。
繼承自java.lang.Exception(java.lang.RuntimeException除外)。
unchecked異常:
表示錯(cuò)誤,程序的邏輯錯(cuò)誤。是RuntimeException的子類,比如IllegalArgumentException, NullPointerException和IllegalStateException。
不需要在代碼中顯式地捕獲unchecked異常做處理。
繼承自java.lang.RuntimeException(而java.lang.RuntimeException繼承自java.lang.Exception)。
看下面的異常結(jié)構(gòu)圖或許層次感更加深些:
三、@Transactional的使用實(shí)例
本實(shí)例采用的是eclipse+maven,maven只是作為jar管理,即便不了解的maven的猿友也可以讀懂。
3.1、spring的配置文件
里面必須先配置tx名字空間如下:
為了使用基于@Transactional的事務(wù)管理,需要在Spring中進(jìn)行如下的配置:
<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" />
博主的整個(gè)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> <!--要是有多個(gè)配置文件,只需在這里繼續(xù)添加即可 --> </list> </property> </bean> <!-- 配置數(shù)據(jù)源 --> <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" /> <!-- 自動(dòng)掃描了所有的XxxxMapper.xml對(duì)應(yīng)的mapper接口文件,這樣就不用一個(gè)一個(gè)手動(dòng)配置Mpper的映射了,只要Mapper接口類和Mapper映射文件對(duì)應(yīng)起來就可以了。 --> <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> <!-- 自動(dòng)掃描注解的bean --> <context:component-scan base-package="com.luo.service" /> </beans>
3.2、使用@Transactional,在添加用戶實(shí)現(xiàn)類方法加上注解
@Transactional(propagation=Propagation.REQUIRED) public void addUser(User user) { userDao.addUser(user); String string = null; if(string.equals("")) { int i = 0; } }
上面的方法我故意讓其出現(xiàn)空指針異常,會(huì)事物回滾
3.3、運(yùn)行單元測(cè)試類
@Test public void addUserTest(){ User user = new User(); user.setUserName("luoguohui1"); user.setUserPassword("luoguohui1"); userService.addUser(user); }
發(fā)現(xiàn)無法插入進(jìn)去,但是如果把@Transactional去掉,即代碼如下,雖然出現(xiàn)異常,但是數(shù)據(jù)庫中還是有添加對(duì)應(yīng)數(shù)據(jù)的:
3.4、源碼下載
本文最終源碼下載:first_maven_project_jb51.rar
四、Spring中的@Transactional必須要了解的概念
Spring中的@Transactional基于動(dòng)態(tài)代理的機(jī)制,提供了一種透明的事務(wù)管理機(jī)制,方便快捷解決在開發(fā)中碰到的問題。
一般使用是通過如下代碼對(duì)方法或接口或類注釋:
@Transactional(propagation=Propagation.NOT_SUPPORTED)
Propagation支持7種不同的傳播機(jī)制:
REQUIRED:如果存在一個(gè)事務(wù),則支持當(dāng)前事務(wù)。如果沒有事務(wù)則開啟一個(gè)新的事務(wù)。
SUPPORTS: 如果存在一個(gè)事務(wù),支持當(dāng)前事務(wù)。如果沒有事務(wù),則非事務(wù)的執(zhí)行。但是對(duì)于事務(wù)同步的事務(wù)管理器,PROPAGATION_SUPPORTS與不使用事務(wù)有少許不同。
NOT_SUPPORTED:總是非事務(wù)地執(zhí)行,并掛起任何存在的事務(wù)。
REQUIRESNEW:總是開啟一個(gè)新的事務(wù)。如果一個(gè)事務(wù)已經(jīng)存在,則將這個(gè)存在的事務(wù)掛起。
MANDATORY:如果已經(jīng)存在一個(gè)事務(wù),支持當(dāng)前事務(wù)。如果沒有一個(gè)活動(dòng)的事務(wù),則拋出異常。
NEVER:總是非事務(wù)地執(zhí)行,如果存在一個(gè)活動(dòng)事務(wù),則拋出異常
NESTED:如果一個(gè)活動(dòng)的事務(wù)存在,則運(yùn)行在一個(gè)嵌套的事務(wù)中。如果沒有活動(dòng)事務(wù),則按REQUIRED屬性執(zhí)行。
下面是一些需要注意的事項(xiàng),必須必須必須要看,不然遇到各種坑別說博主沒有提醒你哦:
下面是一些需要注意的事項(xiàng),必須必須必須要看,不然遇到各種坑別說博主沒有提醒你哦:
下面是一些需要注意的事項(xiàng),必須必須必須要看,不然遇到各種坑別說博主沒有提醒你哦:
在需要事務(wù)管理的地方加@Transactional 注解。@Transactional 注解可以被應(yīng)用于接口定義和接口方法、類定義和類的 public 方法上。
@Transactional 注解只能應(yīng)用到 public 可見度的方法上。 如果你在 protected、private 或者 package-visible 的方法上使用 @Transactional 注解,它也不會(huì)報(bào)錯(cuò), 但是這個(gè)被注解的方法將不會(huì)展示已配置的事務(wù)設(shè)置。
注意僅僅 @Transactional 注解的出現(xiàn)不足于開啟事務(wù)行為,它僅僅 是一種元數(shù)據(jù)。必須在配置文件中使用配置元素,才真正開啟了事務(wù)行為。
通過 元素的 “proxy-target-class” 屬性值來控制是基于接口的還是基于類的代理被創(chuàng)建。如果 “proxy-target-class” 屬值被設(shè)置為 “true”,那么基于類的代理將起作用(這時(shí)需要CGLIB庫cglib.jar在CLASSPATH中)。如果 “proxy-target-class” 屬值被設(shè)置為 “false” 或者這個(gè)屬性被省略,那么標(biāo)準(zhǔn)的JDK基于接口的代理將起作用。
Spring團(tuán)隊(duì)建議在具體的類(或類的方法)上使用 @Transactional 注解,而不要使用在類所要實(shí)現(xiàn)的任何接口上。在接口上使用 @Transactional 注解,只能當(dāng)你設(shè)置了基于接口的代理時(shí)它才生效。因?yàn)樽⒔馐?不能繼承 的,這就意味著如果正在使用基于類的代理時(shí),那么事務(wù)的設(shè)置將不能被基于類的代理所識(shí)別,而且對(duì)象也將不會(huì)被事務(wù)代理所包裝。
@Transactional 的事務(wù)開啟 ,或者是基于接口的 或者是基于類的代理被創(chuàng)建。所以在同一個(gè)類中一個(gè)方法調(diào)用另一個(gè)方法有事務(wù)的方法,事務(wù)是不會(huì)起作用的。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
利用AOP實(shí)現(xiàn)系統(tǒng)告警的方法詳解
在開發(fā)的過程中會(huì)遇到各種各樣的開發(fā)問題,服務(wù)器宕機(jī)、網(wǎng)絡(luò)抖動(dòng)、代碼本身的bug等等。針對(duì)代碼的bug,我們可以提前預(yù)支,通過發(fā)送告警信息來警示我們?nèi)ジ深A(yù),盡早處理。本文將利用AOP實(shí)現(xiàn)系統(tǒng)告警,需要的可以參考一下2022-09-09Jackson2的JsonSchema實(shí)現(xiàn)java實(shí)體類生成json方式
這篇文章主要介紹了Jackson2的JsonSchema實(shí)現(xiàn)java實(shí)體類生成json,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11Java中將一個(gè)列表拆分為多個(gè)較小列表的三種不同方法
有時(shí)候我們需要將大集合按指定的數(shù)量分割成若干個(gè)小集合,這篇文章主要給大家介紹了關(guān)于Java中將一個(gè)列表拆分為多個(gè)較小列表的三種不同方法,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-09-09MyBatis工廠類封裝與簡(jiǎn)化實(shí)現(xiàn)
工廠類的目的是將對(duì)象的創(chuàng)建邏輯封裝在一個(gè)類中,以便客戶端代碼無需了解具體的實(shí)現(xiàn)細(xì)節(jié),本文主要介紹了MyBatis工廠類封裝與簡(jiǎn)化實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下2024-01-01Idea安裝及涉及springboot詳細(xì)配置的圖文教程
這篇文章主要介紹了Idea安裝及涉及springboot詳細(xì)配置,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-10-10java查找字符串中的包含子字符串的個(gè)數(shù)實(shí)現(xiàn)代碼
下面小編就為大家?guī)硪黄猨ava查找字符串中的包含子字符串的個(gè)數(shù)實(shí)現(xiàn)代碼。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-06-06