国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

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

jone5679 / 2209人閱讀

摘要:事務(wù)嵌套影響首先經(jīng)過實驗,結(jié)論一仍然成立,即,當不加上的時候,無論內(nèi)外報,都會回滾無論內(nèi)外報非錯誤,都不會回滾。結(jié)論結(jié)論一對于可以保證錯誤的回滾,如果想保證非錯誤的回滾,需要加上參數(shù)。

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

建兩張表,模擬兩個數(shù)據(jù)操作

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;
測試

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

最簡單測試

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

    @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();
    }

如果事務(wù)內(nèi)報了RuntimeException錯誤,事務(wù)可以回滾。

    @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();
    }

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

    @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 參數(shù)也可以實現(xiàn)回滾。

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

try catch 影響

經(jīng)過博主多種情況測試,發(fā)現(xiàn)try catch對回滾這個事本身沒有什么影響,結(jié)論一照樣成立。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中的錯誤不繼續(xù)網(wǎng)上拋,切面無法感知到錯誤,無法進行處理,那么事務(wù)就無法回滾了。

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

事務(wù)嵌套 影響

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

    @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();
    }

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

下面看情況二,里面的事務(wù)加上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);
    }

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

然后試下里面報錯:

    @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();
    }

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

Propagation propagation() default Propagation.REQUIRED;

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

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

try catch和事務(wù)嵌套 共同影響

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

結(jié)論

結(jié)論一:對于@Transactional可以保證RuntimeException錯誤的回滾,如果想保證非RuntimeException錯誤的回滾,需要加上rollbackFor = Exception.class 參數(shù)。
結(jié)論二:try catch只是對異常是否可以被@Transactional 感知 到有影響。如果錯誤拋到切面可以感知到的地步,那就可以起作用。
結(jié)論三:由于REQUIRED屬性,“兩個事務(wù)”其實是一個事務(wù),處理能力看報錯時刻,是否添加了處理非RuntimeException的能力。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/74647.html

相關(guān)文章

  • 嵌套事務(wù)及事務(wù)不起作用的問題

    摘要:否則事務(wù)不起作用。第二種情況同一個類中方法嵌套方法方法有,方法內(nèi)都沒有,事務(wù)起作用。總結(jié)要想事務(wù)起作用,必須是主方法名上有注解,方法體內(nèi)不能用如果用,則中必須用注解應(yīng)該只被應(yīng)用到方法上,不要用在等方法上,即使用了也將被忽略,不起作用。 代碼方法示例 @Transactional public void saveAA() { ????try { ...

    Blackjun 評論0 收藏0
  • Spring事務(wù)傳播行為詳解

    摘要:前言在接口中規(guī)定了種類型的事務(wù)傳播行為。事務(wù)傳播行為是框架獨有的事務(wù)增強特性,他不屬于的事務(wù)實際提供方數(shù)據(jù)庫行為。本文對七種事務(wù)傳播行為做詳細介紹,內(nèi)容主要代碼示例的方式呈現(xiàn)。 前言 Spring在TransactionDefinition接口中規(guī)定了7種類型的事務(wù)傳播行為。事務(wù)傳播行為是Spring框架獨有的事務(wù)增強特性,他不屬于的事務(wù)實際提供方數(shù)據(jù)庫行為。這是Spring為我們提供...

    cpupro 評論0 收藏0
  • Spring事務(wù)整理

    摘要:使用需要使用作為事務(wù)管理器。兩個事務(wù)互不影響。這是默認的隔離級別,使用數(shù)據(jù)庫默認的事務(wù)隔離級別下邊的四個與的隔離級別相對應(yīng)這是事務(wù)最低的隔離級別,它充許另外一個事務(wù)可以看到這個事務(wù)未提交的數(shù)據(jù)。這種事務(wù)隔離級別可 Spring事務(wù)整理 工作了幾年了,今天抽時間整理一下spring的事務(wù),說起spring的事務(wù)是面試的時候面試官經(jīng)常提及的問題,接下來結(jié)合網(wǎng)上資料再總結(jié)下spring的事務(wù)...

    stackvoid 評論0 收藏0
  • Spring 踩坑之@Transactional 神奇失效

    摘要:引言對于追求數(shù)據(jù)強一致性的系統(tǒng),事務(wù)扮演者十分重要的角色最近在項目中遇到一個事務(wù)失效的問題,在此分享給大家。情景回放問題分析初步分析這是事務(wù)獲取鎖超時導致的錯誤,奇怪的是拋出異常但是事務(wù)沒有回滾。唯一的解釋是事務(wù)失效了。 引言 對于追求數(shù)據(jù)強一致性的系統(tǒng),事務(wù)扮演者十分重要的角色.最近在項目中遇到一個事務(wù)失效的問題,在此分享給大家。 情景回放 ### Cause: com.mysql....

    derek_334892 評論0 收藏0
  • UnexpectedRollbackException解決方案

    摘要:在嵌套事務(wù)場景中,內(nèi)層事務(wù)的和外層事務(wù)的會在外層事務(wù)結(jié)束時進行提交或回滾。解決方案如果希望內(nèi)層事務(wù)拋出異常時中斷程序執(zhí)行,直接在外層事務(wù)的代碼塊中拋出如果希望程序正常執(zhí)行完畢,并且希望外層事務(wù)結(jié)束時全部提交,需要在內(nèi)層事務(wù)中做異常捕獲處理。 前言 最近在項目中發(fā)現(xiàn)了一則報錯:org.springframework.transaction.UnexpectedRollbackExcept...

    mating 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<