日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

spring 中事務注解@Transactional與trycatch的使用

瀏覽:60日期:2023-07-08 11:26:56
spring事務注解@Transactional與trycatch

在項目中 @service層中 我們會經常在做一些增刪改操作的方法上看到 spring 的事務注解 @transaction 已知@transaction 是讓spring 幫我們實現事務的控制。

但是在項目中會經常看到 有的方法中 會存在trycatch塊包括的方法上注解著@transaction

eg:

@Override @Transactional public Json addOrder(TOrderAddReq tOrderAddReq) { try{ //增刪改方法} catch (Exception e) { ..... e.printStackTrace();}//}return json; }

上述的方法執行后可以看到事務并沒有執行,接下來再看一個例子eg:

@Override @Transactional public Json addOrder(TOrderAddReq tOrderAddReq) { try{ //增刪改方法} catch (Exception e) { // 手動硬編碼開啟spring事務管理 TransactionAspectSupport.currentTransactionStatus().setRollbackOnly(); e.printStackTrace();}//}return json; }

上述方法執行后我們可以看到事務最后執行了,但實際上 事務 執行只是因為手動硬編碼開啟spring事務管理起了作用 而方法上的注解并沒有起作用

接下來再看一個例子eg

@Override @Transactional public Json addOrder(TOrderAddReq tOrderAddReq) { try{ //增刪改方法} catch (Exception e) { throw new RuntimeException(); }//}return json; }

上述方法執行后我們可以看到事務是執行了的,但這里有個小細節:@Transactional不做任何配置 默認是對拋出的unchecked異常回滾,checked異常不會回滾,為了讓所有異常都會讓事務啟動可以將 @Transactional配置為 @Transactional(rollbackFor = Exception.class)

解釋:

spring的事務邊界是在調用業務方法之前開始的,業務方法執行完畢之后來執行commit or rollback(spring默認取決于是否拋出runtime異常).

如果拋出runtime exception 并在你的業務方法中沒有catch到的話,事務會回滾。

一般不需要在業務方法中catch異常,如果非要catch,在做完你想做的工作后(比如關閉文件等)一定要拋出runtime exception,否則spring會將你的操作commit,這樣就會產生臟數據.所以你的catch代碼是畫蛇添足。

@Transactional回滾問題(try catch、嵌套)

Spring 事務注解 @Transactional 本來可以保證原子性,如果事務內有報錯的話,整個事務可以保證回滾,但是加上try catch或者事務嵌套,可能會導致事務回滾失敗。測試一波。

準備

建兩張表,模擬兩個數據操作

CREATE TABLE `user` ( `id` int(11) NOT NULL AUTO_INCREMENT, `name` varchar(20) DEFAULT NULL, `age` smallint(3) DEFAULT NULL, PRIMARY KEY (`id`)) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8;CREATE TABLE `role` ( `id` int(11) NOT NULL AUTO_INCREMENT, `role_name` varchar(20) CHARACTER SET utf8 COLLATE utf8_general_ci DEFAULT NULL, PRIMARY KEY (`id`)) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8;測試

根據排列組合原理,我們進行四種測試:1、無try catch、無嵌套;2、有try catch、無嵌套;3、無try catch、有嵌套;4、都有。

最簡單測試

如果我們單純@Transactional,事務可以正常回滾嗎?

@GetMapping('/saveNormal0') @Transactional public void saveNormal0() throws Exception {int age = random.nextInt(100);User user = new User().setAge(age).setName('name:'+age);userService.save(user);throw new RuntimeException(); }

如果事務內報了RuntimeException錯誤,事務可以回滾。

@GetMapping('/saveNormal0') @Transactional public void saveNormal0() throws Exception {int age = random.nextInt(100);User user = new User().setAge(age).setName('name:'+age);userService.save(user);throw new Exception(); }

如果事務內報了Exception錯誤(非RuntimeException錯誤),事務不可以回滾。

@GetMapping('/saveNormal0') @Transactional( rollbackFor = Exception.class) public void saveNormal0() throws Exception {int age = random.nextInt(100);User user = new User().setAge(age).setName('name:'+age);userService.save(user);throw new Exception(); }

如果是Exception錯誤(非RuntimeException),加上 rollbackFor = Exception.class 參數也可以實現回滾。

結論一:對于@Transactional可以保證RuntimeException錯誤的回滾,如果想保證非RuntimeException錯誤的回滾,需要加上rollbackFor = Exception.class 參數。

try catch 影響

經過博主多種情況測試,發現try catch對回滾這個事本身沒有什么影響,結論一照樣成立。try catch只是對異常是否可以被@Transactional 感知 到有影響。如果錯誤拋到切面可以感知到的地步,那就可以起作用。

@GetMapping('/saveTryCatch') @Transactional( rollbackFor = Exception.class) public void saveTryCatch() throws Exception{try{ int age = random.nextInt(100); User user = new User().setAge(age).setName('name:'+age); userService.save(user); throw new Exception();}catch (Exception e){ throw e;} }

比如上面一段代碼就回滾了。

@GetMapping('/saveTryCatch') @Transactional( rollbackFor = Exception.class) public void saveTryCatch() throws Exception{try{ int age = random.nextInt(100); User user = new User().setAge(age).setName('name:'+age); userService.save(user); throw new Exception();}catch (Exception e){} }

然而,將catch中的錯誤不繼續網上拋,切面無法感知到錯誤,無法進行處理,那么事務就無法回滾了。

結論二:try catch只是對異常是否可以被@Transactional 感知 到有影響。如果錯誤拋到切面可以感知到的地步,那就可以起作用。

事務嵌套 影響

首先經過實驗,結論一仍然成立,即,當不加上rollbackFor = Exception.class 的時候,無論內外報RuntimeException,都會回滾;無論內外報 非RuntimeException 錯誤,都不會回滾。如果加上rollbackFor = Exception.class,無論內外怎么報錯,都會回滾。這些代碼就不給出了。接下來,試下下面兩種情況:

@GetMapping('/out') @Transactional( rollbackFor = Exception.class) public void out() throws Exception{innerService.inner();int age = random.nextInt(100);User user = new User().setAge(age).setName('name:' + age);userService.save(user);throw new Exception(); } @Transactional public void inner() throws Exception{Role role = new Role();role.setRoleName('roleName:'+new Random().nextInt(100));roleService.save(role);//throw new Exception(); }

情況一,外面事務加上rollbackFor = Exception.class,里面事務不加,測試內外分別報錯的情況(為了簡化代碼量,只給出了外面報錯的代碼),都可以回滾。因為,無論如何,錯誤都拋給了外面那個事務進行處理,而外面那個加上了rollbackFor = Exception.class,具備處理非RuntimeException錯誤的能力,所以都可以讓事務進行正常回滾。

下面看情況二,里面的事務加上rollbackFor = Exception.class,外面不加,外面報錯。

@GetMapping('/out') @Transactional public void out() throws Exception{innerService.inner();int age = random.nextInt(100);User user = new User().setAge(age).setName('name:' + age);userService.save(user);throw new Exception(); }@Transactional( rollbackFor = Exception.class) public void inner() throws Exception{Role role = new Role();role.setRoleName('roleName:'+new Random().nextInt(100));roleService.save(role); }

事務都無法回滾,這是我們有個疑問,里面的事務明明有很強的處理能力啊,為什么和外面一起回滾失敗呢,別著急,等等聊這個。

然后試下里面報錯:

@GetMapping('/out') @Transactional public void out() throws Exception{innerService.inner();int age = random.nextInt(100);User user = new User().setAge(age).setName('name:' + age);userService.save(user); } @Transactional( rollbackFor = Exception.class) public void inner() throws Exception{Role role = new Role();role.setRoleName('roleName:'+new Random().nextInt(100));roleService.save(role);throw new Exception(); }

咦,這回都進行了正常的回滾。我的天,這回外面沒有處理能力,為什么接受里面拋出來的錯誤,也進行了回滾!!!看上去,就好像里外事務總是同生共死的對不對?原來,@Transactional還有個參數,看下源碼,這個注解還有默認值:

Propagation propagation() default Propagation.REQUIRED;

REQUIRED的意思是說,事務嵌套的時候,如果發現已經有事務存在了,就加入這個事務,而不是新建一個事務,所以根本就不存在兩個事務,一直只有一個!至于,此參數其他值,本文不進行測試。回到上面的問題,當外面報錯的時候,此時查看事務,沒有增加rollbackFor = Exception.class參數,即沒有處理非RuntimeException能力,所以代碼走完,貌似“兩個事務”,都回滾失敗了。當里面報錯的時候,事務已經添加上了處理非RuntimeException能力,所以,代碼走完就回滾成功了。

結論三:由于REQUIRED屬性,“兩個事務”其實是一個事務,處理能力看報錯時刻,是否添加了處理非RuntimeException的能力。

try catch和事務嵌套 共同影響

在結論一二三成立的條件下,探索共同影響的問題就簡單多了,由于情況太多,就不進行過多的代碼展示了。

結論

結論一:

對于@Transactional可以保證RuntimeException錯誤的回滾,如果想保證非RuntimeException錯誤的回滾,需要加上rollbackFor = Exception.class 參數。

結論二:

try catch只是對異常是否可以被@Transactional 感知 到有影響。如果錯誤拋到切面可以感知到的地步,那就可以起作用。

結論三:

由于REQUIRED屬性,“兩個事務”其實是一個事務,處理能力看報錯時刻,是否添加了處理非RuntimeException的能力。

以上為個人經驗,希望能給大家一個參考,也希望大家多多支持好吧啦網。

標簽: Spring
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
91国内精品| 久久一区亚洲| 色吊丝一区二区| 国户精品久久久久久久久久久不卡 | 亚洲香蕉久久| 六月婷婷一区| 91福利精品在线观看| 国产欧美69| 国产欧美日韩一级| 国产精品玖玖玖在线资源| 蜜桃精品视频| 精品国产乱码久久久久久樱花| 日韩av一区二区三区四区| 久久精品超碰| 成人日韩av| 亚洲一区二区三区高清| 日韩av午夜在线观看| 麻豆精品少妇| 91精品二区| 欧美另类中文字幕| 精品国产乱码久久久久久樱花| 亚洲黄色免费av| 久久国产99| 久久国产麻豆精品| av资源中文在线天堂| 欧美精品自拍| 国产精品久久久久毛片大屁完整版| 亚洲啊v在线| 日韩欧美中文字幕电影| 国产不卡一区| 天堂成人免费av电影一区| 日韩成人午夜精品| 国产欧洲在线| 日韩一区二区三区高清在线观看| 成人在线黄色| 欧美在线看片| 天使萌一区二区三区免费观看| 精品淫伦v久久水蜜桃| 亚洲一二三区视频| 国产精品分类| 女同性一区二区三区人了人一| 国产精品最新| 亚洲久久视频| 日韩欧美一区二区三区免费观看| 首页亚洲欧美制服丝腿| 欧美不卡高清一区二区三区| 日韩一区二区三区在线看| 免费毛片在线不卡| 婷婷激情一区| 国产精品密蕾丝视频下载| 国产视频欧美| 午夜精品久久久久久久久久蜜桃| 亚洲精品在线国产| 久久精品高清| 美女精品一区二区| 国产精品成人国产| 日韩高清二区| 91精品国产经典在线观看| 亚洲麻豆一区| 日韩一区二区三区在线看| 免费精品视频| 亚洲在线久久| 日韩中文字幕| 久久国产精品免费精品3p| 国产精品毛片久久久| 精品视频一区二区三区四区五区 | 丝袜美腿诱惑一区二区三区| 成人国产精品久久| 精精国产xxxx视频在线野外| 国产91一区| 亚洲免费福利一区| 国产私拍福利精品视频二区| 欧美1区二区| 久草免费在线视频| 亚洲国产专区校园欧美| 蜜臀av性久久久久蜜臀aⅴ四虎| 丝袜美腿亚洲一区| 欧美片第1页综合| 视频在线不卡免费观看| 不卡中文字幕| 日本va欧美va精品发布| 国产高潮在线| 日韩一区欧美二区| 精品视频亚洲| 美国欧美日韩国产在线播放| 国产亚洲电影| 不卡一区2区| 狂野欧美性猛交xxxx| 欧美丝袜一区| 国产精品tv| 免费日韩av片| 国产剧情在线观看一区| 久久九九精品| 国产精品一区二区99| 日韩一区二区久久| 韩国一区二区三区视频| 最新国产精品久久久| 四虎成人av| 国产精品中文字幕亚洲欧美| 91高清一区| 欧美男人天堂| 国产精品一区三区在线观看| 国产手机视频一区二区| 91视频久久| 国产精品一二| 综合一区在线| 欧美日韩国产高清| 狠狠久久伊人| 久久精品女人| 美女视频黄 久久| 日本欧美久久久久免费播放网| 午夜精品婷婷| 亚洲福利免费| 久久中文字幕二区| 日韩中文字幕高清在线观看| 欧美在线观看天堂一区二区三区| 鲁大师影院一区二区三区| 伊人久久高清| 久久九九国产| 九一精品国产| 夜夜嗨av一区二区三区网站四季av| 精品日产乱码久久久久久仙踪林| 日韩欧乱色一区二区三区在线| 日韩精品一二三| 日韩精品一卡二卡三卡四卡无卡 | 欧美一级二级三级视频| 91成人精品在线| 国产亚洲久久| 六月丁香综合在线视频| 精品三级在线观看视频| 国产一区调教| 欧美日韩在线二区| 国产精品老牛| 日韩有码av| 国产精品中文字幕制服诱惑| 日韩一区二区三区精品视频第3页| 蜜桃av一区二区三区电影| 婷婷综合福利| 美女在线视频一区| 日韩在线精品| 鲁大师成人一区二区三区| 亚洲人亚洲人色久| 国产精品久久久久久模特| 成人综合一区| 在线成人直播| 久久国产尿小便嘘嘘| 国产精品精品国产一区二区| 欧美二区视频| 欧美一级网址| 国产99精品| 亚洲精品看片| 丁香六月综合| 视频国产精品| 美女福利一区二区三区| 男人的天堂亚洲一区| 国产精品一国产精品k频道56| 人人草在线视频| 亚洲伊人精品酒店| 国产精品福利在线观看播放| 亚洲免费播放| 91日韩免费| 日韩成人一级| 妖精视频成人观看www| 久久精品福利| 日韩精品一级| 欧美a级片一区| 美女视频网站久久| 免费日韩精品中文字幕视频在线| 国产+成+人+亚洲欧洲在线| 综合欧美亚洲| 中文欧美日韩| 日韩欧美一区二区三区在线观看 | 午夜精品免费| 成人午夜在线| 日韩欧美三区| 亚洲伊人精品酒店| 国内精品福利| av资源中文在线| 国产一区二区三区成人欧美日韩在线观看| 久久香蕉精品| 久久午夜影视| 亚洲欧美视频| 制服诱惑一区二区| 国精品一区二区三区| 亚洲成人av观看| 狠狠色狠狠色综合日日tαg| 色综合www| 成人精品天堂一区二区三区| 久久精品午夜| 国产精品一区二区精品视频观看| 中文字幕成人| 日本欧美一区二区| 欧美一区=区三区| 久久国内精品自在自线400部| 91在线成人| 国产精品资源| av中文字幕在线观看第一页| 樱桃视频成人在线观看| 久久婷婷亚洲| 激情91久久|