国产一区二区精品-国产一区二区精品久-国产一区二区精品久久-国产一区二区精品久久91-免费毛片播放-免费毛片基地

千鋒教育-做有情懷、有良心、有品質(zhì)的職業(yè)教育機(jī)構(gòu)

手機(jī)站
千鋒教育

千鋒學(xué)習(xí)站 | 隨時(shí)隨地免費(fèi)學(xué)

千鋒教育

掃一掃進(jìn)入千鋒手機(jī)站

領(lǐng)取全套視頻
千鋒教育

關(guān)注千鋒學(xué)習(xí)站小程序
隨時(shí)隨地免費(fèi)學(xué)習(xí)課程

當(dāng)前位置:首頁  >  技術(shù)干貨  > 詳解rollbackfor

詳解rollbackfor

來源:千鋒教育
發(fā)布人:xqq
時(shí)間: 2023-11-23 00:32:00 1700670720

在編寫Java應(yīng)用程序時(shí),可以使用Spring中的事務(wù)管理來控制數(shù)據(jù)庫事務(wù)的邊界。在這種情況下,我們需要考慮異常處理的問題。Spring為我們提供了非常簡(jiǎn)單和方便的方法來處理異常,在這篇文章中,我們將要詳細(xì)介紹rollbackfor。這是Spring事務(wù)管理中一個(gè)非常重要的屬性,我們來了解一下。

一、rollbackfor多個(gè)異常

默認(rèn)情況下,Spring的事務(wù)管理器只會(huì)在出現(xiàn)運(yùn)行時(shí)異常時(shí)回滾事務(wù),但是我們也可以使用rollbackfor屬性來指定需要回滾的異常。我們可以指定一個(gè)或多個(gè)異常類型,讓事務(wù)管理器在這些異常出現(xiàn)時(shí)回滾事務(wù)。下面的代碼演示了如何處理多個(gè)異常類型。


@Transactional(rollbackFor = {SQLException.class, IOException.class})
public void updateCustomer(Customer customer) throws SQLException, IOException {
    // ...
}

在上面的代碼中,如果在執(zhí)行方法時(shí)發(fā)生了SQLException或IOException異常,事務(wù)管理器將回滾事務(wù)。

二、rollbackfor事務(wù)級(jí)別

rollbackfor屬性定義了需要回滾的異常類型,但是它也有一個(gè)事務(wù)級(jí)別的概念。如果在嵌套的事務(wù)中發(fā)生了指定的異常類型,只有在最外層的事務(wù)中才會(huì)回滾事務(wù)。如果沒有最外層的事務(wù),異常的行為將取決于rollbackfor屬性的值。下面的例子展示了當(dāng)存在嵌套事務(wù)時(shí),rollbackfor屬性是如何工作的:


@Transactional
public void purchaseProduct(String productId, int quantity) {
    try {
        // 嵌套事務(wù)
        orderService.createOrder(productId, quantity);
    } catch (SQLException ex) {
        // 嵌套事務(wù)中的SQLException將不會(huì)回滾整個(gè)事務(wù)
        // ...
    }
}

在上面的代碼中,方法purchaseProduct()是一個(gè)事務(wù)方法。但是,createOrder()方法也是一個(gè)事務(wù)方法,并且通過try-catch塊捕獲了SQLException異常。如果createOrder()方法中拋出了一個(gè)SQLException異常,只有其自己的事務(wù)被回滾。因此,在purchaseProduct()方法中的事務(wù)仍然會(huì)提交。但是Spring提供了一個(gè)更好的方法來解決這個(gè)問題:使用Propagation.MANDATORY。改變createOrder()方法的事務(wù)定義,如下所示:


@Transactional(propagation = Propagation.MANDATORY, rollbackFor = SQLException.class)
public void createOrder(String productId, int quantity) throws SQLException {
    // ...
}

在上面的代碼中,Propagation.MANDATORY指定了createOrder()方法需要在已經(jīng)存在的事務(wù)中運(yùn)行。如果沒有已經(jīng)存在的事務(wù),將會(huì)拋出異常。此外,這里也指定了SQLException異常需要回滾整個(gè)事務(wù)。

三、rollbackfor與rollbackon

當(dāng)需要回滾多個(gè)異常類型時(shí),我們也可以使用rollbackon屬性。它只是rollbackfor屬性的簡(jiǎn)化版,允許我們回滾單個(gè)異常。同時(shí),如果我們指定了rollbackon和rollbackfor屬性,當(dāng)出現(xiàn)任何一個(gè)滿足條件的異常時(shí),都會(huì)回滾事務(wù)。下面的代碼演示了如何定義rollbackon屬性:


@Transactional(rollbackOn = IOException.class)
public void updateCustomer(Customer customer) throws SQLException, IOException {
    // ...
}

在上面的代碼中,如果執(zhí)行方法時(shí)拋出了IOException異常,事務(wù)將被回滾。

四、rollbackfor屬性

rollbackfor屬性是@Transactional注解中的一個(gè)屬性,它定義了需要回滾的異常。我們可以為該屬性指定一個(gè)或多個(gè)異常類,只有指定的異常類出現(xiàn)時(shí),事務(wù)才會(huì)回滾。如果沒有指定該屬性,則默認(rèn)只有RuntimeException的子類異常會(huì)回滾。如果指定了這個(gè)屬性,就不需要使用rollbackon屬性。


@Transactional(rollbackFor = Exception.class)
public void updateCustomer(Customer customer) throws SQLException, IOException {
    // ...
}

在上面的代碼中,如果執(zhí)行方法時(shí)拋出了Exception或其子類異常,事務(wù)將被回滾。

五、rollbackfor沒有回滾

在某些情況下,即使我們指定了rollbackfor屬性,事務(wù)也不會(huì)回滾。這可能是由于以下原因之一:

回滾的異常類型不被拋出 異常沒有被捕獲 異常沒有被正常拋出

下面的代碼演示了一個(gè)這樣的例子:


@Transactional(rollbackFor = SQLException.class)
public void updateCustomer(Customer customer) throws SQLException {
    try{
        // ...
        throw new IOException("Simulate an IOException");
    } catch (IOException ex) {
        // ...
    }
}

在上面的代碼中,我們指定了rollbackfor屬性以回滾SQLException異常。但是,在try塊中引發(fā)了IOException異常。由于該異常沒有被拋出,所以事務(wù)不會(huì)被回滾。

六、rollbackfor默認(rèn)值

如果我們使用@Transactional注解而沒有定義rollbackfor屬性,則默認(rèn)的回滾異常只有RuntimeException及其子類異常。如果定義了rollbackfor屬性,則將會(huì)覆蓋這個(gè)默認(rèn)值。

七、rollbackfor不能回滾

有些情況下,即使使用rollbackfor屬性,事務(wù)也不能回滾。這可能是由于以下原因之一:

當(dāng)前的方法不是在事務(wù)中執(zhí)行的 事務(wù)被另一個(gè)事務(wù)管理器控制 在try塊中拋出的異常被捕獲并處理了

下面的代碼演示了一個(gè)這樣的例子:


public void updateCustomer(Customer customer) throws SQLException {
    try {
        transactionTemplate.execute(new TransactionCallbackWithoutResult() {
            protected void doInTransactionWithoutResult(TransactionStatus status) {
                // ...
                throw new SQLException("Simulate a SQLException");
            }
        });
    } catch (Exception ex) {
        // ...
    }
}

在上面的代碼中,我們使用了TransactionTemplate來執(zhí)行一個(gè)事務(wù),并且在該事務(wù)中拋出了一個(gè)SQLException異常。但是,該異常被捕獲并在catch塊中處理了,所以該事務(wù)不會(huì)被回滾。

八、rollbackfor Throwable

在一些特殊情況下,我們可能需要回滾所有異常,包括Throwable及其所有子類異常。為此,我們可以將rollbackfor屬性設(shè)置為Throwable.class。下面的代碼演示了如何回滾所有異常:


@Transactional(rollbackFor = Throwable.class)
public void updateCustomer(Customer customer) throws SQLException, IOException {
    // ...
}

在上面的代碼中,所有的異常,包括Throwable及其所有子類異常,都將引發(fā)事務(wù)回滾。

九、rollbackforclassname

在某些情況下,我們需要?jiǎng)討B(tài)指定需要回滾的異常類型。為了實(shí)現(xiàn)這個(gè)目的,我們可以使用rollbackforclassname屬性。該屬性是一個(gè)字符串?dāng)?shù)組,用于指定需要回滾的異常類型的完整類名。下面的代碼演示了如何使用rollbackforclassname屬性:


@Transactional(rollbackForClassName = {"java.sql.SQLException", "java.io.IOException"})
public void updateCustomer(Customer customer) throws SQLException, IOException {
    // ...
}

在上面的代碼中,我們通過rollbackforclassname屬性指定了需要回滾的異常類型。這些異常類型的完整類名包括java.sql.SQLException和java.io.IOException。

總結(jié)

在本文中,我們對(duì)rollbackfor屬性進(jìn)行了詳細(xì)介紹。rollbackfor屬性是Spring事務(wù)管理中一個(gè)非常重要的屬性,它允許我們指定需要回滾的異常類型。我們從多個(gè)方面講解了rollbackfor的使用方法,包括回滾多個(gè)異常、事務(wù)級(jí)別、與rollbackon的區(qū)別、默認(rèn)值、不能回滾的情況、Throwable類型和rollbackforclassname屬性。希望這篇文章可以幫助您更好地掌握rollbackfor屬性,并在實(shí)際編程中得到應(yīng)用。

tags: rollbackfor
聲明:本站稿件版權(quán)均屬千鋒教育所有,未經(jīng)許可不得擅自轉(zhuǎn)載。
10年以上業(yè)內(nèi)強(qiáng)師集結(jié),手把手帶你蛻變精英
請(qǐng)您保持通訊暢通,專屬學(xué)習(xí)老師24小時(shí)內(nèi)將與您1V1溝通
免費(fèi)領(lǐng)取
今日已有369人領(lǐng)取成功
劉同學(xué) 138****2860 剛剛成功領(lǐng)取
王同學(xué) 131****2015 剛剛成功領(lǐng)取
張同學(xué) 133****4652 剛剛成功領(lǐng)取
李同學(xué) 135****8607 剛剛成功領(lǐng)取
楊同學(xué) 132****5667 剛剛成功領(lǐng)取
岳同學(xué) 134****6652 剛剛成功領(lǐng)取
梁同學(xué) 157****2950 剛剛成功領(lǐng)取
劉同學(xué) 189****1015 剛剛成功領(lǐng)取
張同學(xué) 155****4678 剛剛成功領(lǐng)取
鄒同學(xué) 139****2907 剛剛成功領(lǐng)取
董同學(xué) 138****2867 剛剛成功領(lǐng)取
周同學(xué) 136****3602 剛剛成功領(lǐng)取
相關(guān)推薦HOT