欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring中的10種事務失效的常見場景

 更新時間:2023年11月09日 10:11:39   作者:莫輕言舞  
這篇文章主要介紹了Spring中的10種事務失效的常見場景,Spring的聲明式事務功能更是提供了極其方便的事務配置方式,配合Spring Boot的自動配置,大多數(shù)Spring Boot項目只需要在方法上標記@Transactional注解,即可一鍵開啟方法的事務性配置,需要的朋友可以參考下

Spring事務失效

Spring針對Java Transaction API (JTA)、JDBC、Hibernate和Java Persistence API(JPA)等事務 API,實現(xiàn)了一致的編程模型,而Spring的聲明式事務功能更是提供了極其方便的事務配置方式,配合Spring Boot的自動配置,大多數(shù)Spring Boot項目只需要在方法上標記@Transactional注解,即可一鍵開啟方法的事務性配置。

但是,事務如果沒有被正確出,很有可能會導致事務的失效,帶來意想不到的數(shù)據(jù)不一致問題,隨后就是大量的人工接入查看和修復數(shù)據(jù),該篇主要分享Spring事務在技術上的正確使用方式,避免因為事務處理不當導致業(yè)務邏輯產(chǎn)生大量偶發(fā)性BUG。

在分析事務失效的常見場景之前,我們先來了解一下:事務的傳播類型 和 @Transactionnal 注解的不同屬性的含義。

事務的傳播類型

//如果有事務, 那么加入事務, 沒有的話新建一個(默認)
@Transactional(propagation=Propagation.REQUIRED)
//容器不為這個方法開啟事務 
@Transactional(propagation=Propagation.NOT_SUPPORTED)
//不管是否存在事務, 都創(chuàng)建一個新的事務, 原來的掛起, 新的執(zhí)行完畢, 繼續(xù)執(zhí)行老的事務 
@Transactional(propagation=Propagation.REQUIRES_NEW) 
//必須在一個已有的事務中執(zhí)行, 否則拋出異常
@Transactional(propagation=Propagation.MANDATORY) 
//必須在一個沒有的事務中執(zhí)行, 否則拋出異常(與Propagation.MANDATORY相反)
@Transactional(propagation=Propagation.NEVER) 
//如果其他bean調用這個方法, 在其他bean中聲明事務, 那就用事務, 如果其他bean沒有聲明事務, 那就不用事務
@Transactional(propagation=Propagation.SUPPORTS) 

isolation

該屬性用于設置底層數(shù)據(jù)庫的事務隔離級別,事務的隔離級別介紹:

// 讀取未提交數(shù)據(jù)(會出現(xiàn)臟讀, 不可重復讀) 基本不使用
@Transactional(isolation = Isolation.READ_UNCOMMITTED)
// 讀取已提交數(shù)據(jù)(會出現(xiàn)不可重復讀和幻讀) Oracle默認
@Transactional(isolation = Isolation.READ_COMMITTED)
// 可重復讀(會出現(xiàn)幻讀) MySQL默認
@Transactional(isolation = Isolation.REPEATABLE_READ)
// 串行化
@Transactional(isolation = Isolation.SERIALIZABLE)

@Transactionnal注解屬性

@Transactional注解可以作用于接口、接口方法、類以及類方法上,它可以通過不同的參數(shù)來選擇什么類型Exception異常下執(zhí)行回滾或者不回滾操作。

Spring事務失效的場景

1. 事務方法未被Spring管理

如果事務方法所在的類沒有注冊到Spring IOC容器中,也就是說,事務方法所在類并沒有被Spring管理,則Spring事務會失效,舉個例子:

/**
 * 商品業(yè)務實現(xiàn)層
 *
 * @author: austin
 * @since: 2023/2/10 14:19
 */
public class ProductServiceImpl extends ServiceImpl<ProductMapper, Product> implements IProductService {
 
    @Autowired
    private ProductMapper productMapper;
 
    @Override
    @Transactional(propagation = Propagation.REQUIRES_NEW)
    public void updateProductStockById(Integer stockCount, Long productId) {
        productMapper.updateProductStockById(stockCount, productId);
    }
}

ProductServiceImpl實現(xiàn)類上沒有添加@Service注解,Product的實例也就沒有被加載到Spring IOC容器,此時updateProductStockById()方法的事務就會在Spring中失效。

2. 方法使用final類型修飾

有時候,某個方法不想被子類重新,這時可以將該方法定義成final的。普通方法這樣定義是沒問題的,但如果將事務方法定義成final,例如:

@Service
public class OrderServiceImpl {
 
    @Transactional
    public final void cancel(OrderDTO orderDTO) {
        // 取消訂單
        cancelOrder(orderDTO);
    }
}

OrderServiceImpl的cancel取消訂單方法被final修飾符修飾,Spring事務底層使用了AOP,也就是通過JDK動態(tài)代理或者cglib,幫我們生成了代理類,在代理類中實現(xiàn)的事務功能。但如果某個方法用final修飾了,那么在它的代理類中,就無法重寫該方法,從而無法添加事務功能。這種情況事務就會在Spring中失效。

Tips: 如果某個方法是static的,同樣無法通過動態(tài)代理將方法聲明為事務方法。

3. 非public修飾的方法

如果事務方式不是public修飾,此時Spring事務會失效,舉個例子:

/**
 * 商品業(yè)務實現(xiàn)層
 *
 * @author: austin
 * @since: 2023/2/10 14:19
 */
@Service
public class ProductServiceImpl extends ServiceImpl<ProductMapper, Product> implements IProductService {
 
    @Autowired
    private ProductMapper productMapper;
 
    @Override
    @Transactional(propagation = Propagation.REQUIRES_NEW)
    private void updateProductStockById(Integer stockCount, String productId) {
        productMapper.updateProductStockById(stockCount, productId);
    }
}

雖然ProductServiceImpl添加了@Service注解,同時updateProductStockById()方法上添加了@Transactional(propagation = Propagation.REQUIRES_NEW)注解,但是由于事務方法updateProductStockById()被 private 定義為方法內私有,同樣Spring事務會失效。

4. 同一個類中的方法相互調用

@Service
public class OrderServiceImpl extends ServiceImpl<OrderMapper, Order> implements IOrderService {
    @Autowired
    private OrderMapper orderMapper;
    @Autowired
    private ProductMapper productMapper;
 
    @Override
    public ResponseEntity submitOrder(Order order) {
        // 保存生成訂單信息
        long orderNo = Math.abs(ThreadLocalRandom.current().nextLong(1000));
        order.setOrderNo("ORDER_" + orderNo);
        orderMapper.insert(order);
 
        // 扣減庫存
        this.updateProductStockById(order.getProductId(), 1L);
        return new ResponseEntity(HttpStatus.OK);
    }
 
    @Transactional(propagation = Propagation.REQUIRES_NEW)
    public void updateProductStockById(Integer num, Long productId) {
        productMapper.updateProductStockById(num, productId);
    }
}

submitOrder()方法和updateProductStockById()方法都在OrderService類中,然而submitOrder()方法沒有添加事務注解,updateProductStockById()方法雖然添加了事務注解,這種情況updateProductStockById()會在Spring事務中失效。

5. 方法的事務傳播類型不支持事務

如果內部方法的事務傳播類型為不支持事務的傳播類型,則內部方法的事務同樣會在Spring中失效,舉個例子:

@Service
public class OrderServiceImpl extends ServiceImpl<OrderMapper, Order> implements IOrderService {
    @Autowired
    private OrderMapper orderMapper;
    @Autowired
    private ProductMapper productMapper;
 
    @Override
    @Transactional(propagation = Propagation.REQUIRES_NEW)
    public ResponseEntity submitOrder(Order order) {
        long orderNo = Math.abs(ThreadLocalRandom.current().nextLong(1000));
        order.setOrderNo("ORDER_" + orderNo);
        orderMapper.insert(order);
 
        // 扣減庫存
        this.updateProductStockById(order.getProductId(), 1L);
        return new ResponseEntity(HttpStatus.OK);
    }
 
 
    /**
     * 扣減庫存方法事務類型聲明為NOT_SUPPORTED不支持事務的傳播
     */
    @Transactional(propagation = Propagation.NOT_SUPPORTED)
    public void updateProductStockById(Integer num, Long productId) {
        productMapper.updateProductStockById(num, productId);
    }
}

6. 異常被內部catch,程序生吞異常

@Service
public class OrderServiceImpl extends ServiceImpl<OrderMapper, Order> implements IOrderService {
    @Autowired
    private OrderMapper orderMapper;
    @Autowired
    private ProductMapper productMapper;
 
    @Override
    @Transactional(propagation = Propagation.REQUIRES_NEW)
    public ResponseEntity submitOrder(Order order) {
        long orderNo = Math.abs(ThreadLocalRandom.current().nextLong(1000));
        order.setOrderNo("ORDER_" + orderNo);
        orderMapper.insert(order);
 
        // 扣減庫存
        this.updateProductStockById(order.getProductId(), 1L);
        return new ResponseEntity(HttpStatus.OK);
    }
 
    /**
     * 扣減庫存方法事務類型聲明為NOT_SUPPORTED不支持事務的傳播
     */
    @Transactional(propagation = Propagation.NOT_SUPPORTED)
    public void updateProductStockById(Integer num, Long productId) {
        try {
            productMapper.updateProductStockById(num, productId);
        } catch (Exception e) {
            // 這里僅僅是捕獲異常之后的打?。ㄏ喈斢诔绦蛲痰袅水惓#?
            log.error("Error updating product Stock: {}", e);
        }
    }
}

7. 數(shù)據(jù)庫不支持事務

Spring事務生效的前提是連接的數(shù)據(jù)庫支持事務,如果底層的數(shù)據(jù)庫都不支持事務,則Spring事務肯定會失效的,例如:使用MySQL數(shù)據(jù)庫,選用MyISAM存儲引擎,因為MyISAM存儲引擎本身不支持事務,因此事務毫無疑問會失效。

8. 未配置開啟事務

如果項目中沒有配置Spring的事務管理器,即使使用了Spring的事務管理功能,Spring的事務也不會生效,例如,如果你是Spring Boot項目,沒有在SpringBoot項目中配置如下代碼:

@Bean
public PlatformTransactionManager transactionManager(DataSource dataSource) {
    return new DataSourceTransactionManager(dataSource);
}

如果是以往的Spring MVC項目,如果沒有配置下面的代碼,Spring事務也不會生效,正常需要在applicationContext.xml文件中,手動配置事務相關參數(shù),比如:

<!-- 配置事務管理器 --> 
<bean class="org.springframework.jdbc.datasource.DataSourceTransactionManager" id="transactionManager"> 
    <property name="dataSource" ref="dataSource"></property> 
</bean> 
<tx:advice id="advice" transaction-manager="transactionManager"> 
    <tx:attributes> 
        <tx:method name="*" propagation="REQUIRED"/>
    </tx:attributes> 
</tx:advice> 
<!-- 用切點把事務切進去 --> 
<aop:config> 
    <aop:pointcut expression="execution(* com.universal.ubdk.*.*(..))" id="pointcut"/> 
    <aop:advisor advice-ref="advice" pointcut-ref="pointcut"/> 
</aop:config> 

9. 錯誤的傳播特性

其實,我們在使用@Transactional注解時,是可以指定propagation參數(shù)的。

該參數(shù)的作用是指定事務的傳播特性,目前Spring支持7種傳播特性:

  1. REQUIRED 如果當前上下文中存在事務,那么加入該事務,如果不存在事務,創(chuàng)建一個事務,這是默認的傳播屬性值。
  2. SUPPORTS 如果當前上下文存在事務,則支持事務加入事務,如果不存在事務,則使用非事務的方式執(zhí)行。
  3. MANDATORY 如果當前上下文中存在事務,否則拋出異常。
  4. REQUIRES_NEW 每次都會新建一個事務,并且同時將上下文中的事務掛起,執(zhí)行當前新建事務完成以后,上下文事務恢復再執(zhí)行。
  5. NOT_SUPPORTED 如果當前上下文中存在事務,則掛起當前事務,然后新的方法在沒有事務的環(huán)境中執(zhí)行。
  6. NEVER 如果當前上下文中存在事務,則拋出異常,否則在無事務環(huán)境上執(zhí)行代碼。
  7. NESTED 如果當前上下文中存在事務,則嵌套事務執(zhí)行,如果不存在事務,則新建事務。

如果我們在手動設置propagation參數(shù)的時候,把傳播特性設置錯了,比如:

@Service
public class OrderServiceImpl {
 
    @Transactional(propagation = Propagation.NEVER)
    public void cancelOrder(UserModel userModel) {
        // 取消訂單
        cancelOrder(orderDTO);
        // 還原庫存
        restoreProductStock(orderDTO.getProductId(), orderDTO.getProductCount());
    }
}

我們可以看到cancelOrder()方法的事務傳播特性定義成了Propagation.NEVER,這種類型的傳播特性不支持事務,如果有事務則會拋異常。

10. 多線程調用

在實際項目開發(fā)中,多線程的使用場景還是挺多的。如果Spring事務用在多線程場景中使用不當,也會導致事務無法生效。

@Slf4j
@Service
public class OrderServiceImpl {
 
    @Autowired
    private OrderMapper orderMapper;
    @Autowired
    private MessageService messageService;
 
    @Transactional
    public void orderCommit(orderModel orderModel) throws Exception {
        orderMapper.insertOrder(orderModel);
        new Thread(() -> {
            messageService.sendSms();
        }).start();
    }
}
 
@Service
public class MessageService {
 
    @Transactional
    public void sendSms() {
        // 發(fā)送短信
    }
}

通過示例,我們可以看到訂單提交的事務方法orderCommit()中,調用了發(fā)送短信的事務方法sendSms(),但是發(fā)送短信的事務方法sendSms()是另起了一個線程調用的。

這樣會導致兩個方法不在同一個線程中,從而是兩個不同的事務。如果是sendSms()方法中拋了異常,orderCommit()方法也回滾是不可能的。

實際上,Spring的事務是通過ThreadLocal來保證線程安全的,事務和當前線程綁定,多個線程自然會讓事務失效。

總結

本篇文章主要是介紹Spring事務傳播特性,闡明了@Transactional注解屬性的使用方式,通過不同的代碼示例演示了Spring事務失效的常見場景

到此這篇關于Spring中的10種事務失效的常見場景的文章就介紹到這了,更多相關Spring常見的事務失效內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • java實現(xiàn)雷霆戰(zhàn)機

    java實現(xiàn)雷霆戰(zhàn)機

    這篇文章主要為大家詳細介紹了java實現(xiàn)雷霆戰(zhàn)機,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-06-06
  • java字符串替換排序實例

    java字符串替換排序實例

    這篇文章主要介紹了java字符串替換排序實例,有需要的朋友可以參考一下
    2014-01-01
  • EasyCode插件使用詳解(推薦)

    EasyCode插件使用詳解(推薦)

    EasyCode是idea的一個插件,這個插件功能很強大,今天通過本文給大家分享EasyCode插件使用詳解,需要的朋友可以參考下
    2020-09-09
  • 深入理解Java設計模式之單例模式

    深入理解Java設計模式之單例模式

    這篇文章主要介紹了JAVA設計模式之單例模式的的相關資料,文中示例代碼非常詳細,供大家參考和學習,感興趣的朋友可以了解下
    2021-11-11
  • Java讀寫txt文件代碼實例

    Java讀寫txt文件代碼實例

    這篇文章主要給大家介紹了關于Java讀寫txt文件的相關資料,近期處理的數(shù)據(jù)規(guī)模比較大,正好又是統(tǒng)計合并的事情,想著借助excel就可以完成了,然后就了解了下java讀取excel的事情,需要的朋友可以參考下
    2023-09-09
  • SpringBoot詳解實現(xiàn)自定義異常處理頁面方法

    SpringBoot詳解實現(xiàn)自定義異常處理頁面方法

    SpringBoot是Spring全家桶的成員之一,是一種整合Spring技術棧的方式(或者說是框架),同時也是簡化Spring的一種快速開發(fā)的腳手架
    2022-06-06
  • java中Hibernate面試知識點整理

    java中Hibernate面試知識點整理

    在本篇文章里小編給大家整理的是一篇關于java中Hibernate面試知識點整理內容,有興趣的朋友們可以學習參考下。
    2021-01-01
  • java 壓縮和解壓縮Zip、Jar、Gzip文件實例代碼

    java 壓縮和解壓縮Zip、Jar、Gzip文件實例代碼

    本文主要介紹java壓縮和解壓縮Zip、Jar、Gzip文件的知識,這里整理了相關資料,并附示例代碼有興趣的小伙伴可以參考下
    2016-09-09
  • java 多態(tài)性詳解及常見面試題

    java 多態(tài)性詳解及常見面試題

    這篇文章主要介紹了java 多態(tài)性詳解及常見面試題的相關資料,這里對java 的多態(tài)性做了資料整理,并列舉常見的關于多態(tài)性的面試題,需要的朋友可以參考下
    2016-11-11
  • springBoot如何動態(tài)加載資源文件

    springBoot如何動態(tài)加載資源文件

    這篇文章主要介紹了springBoot如何動態(tài)加載資源文件,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-12-12

最新評論