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

Spring事務(wù)失效的場(chǎng)景梳理總結(jié)

 更新時(shí)間:2023年02月23日 09:22:46   作者:天空~華  
實(shí)際項(xiàng)目開(kāi)發(fā)中,如果涉及到多張表操作時(shí),為了保證業(yè)務(wù)數(shù)據(jù)的一致性,大家一般都會(huì)采用事務(wù)機(jī)制,好多小伙伴可能只是簡(jiǎn)單了解一下,遇到事務(wù)失效的情況,便會(huì)無(wú)從下手,下面這篇文章主要給大家介紹了關(guān)于Spring事務(wù)失效場(chǎng)景的相關(guān)資料,需要的朋友可以參考下

概述

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

但是,事務(wù)如果沒(méi)有被正確使用,很有可能會(huì)導(dǎo)致事務(wù)的失效,帶來(lái)意想不到的數(shù)據(jù)不一致問(wèn)題,隨后就是大量的人工查找問(wèn)題和修復(fù)數(shù)據(jù),本次主要分享Spring事務(wù)在技術(shù)上的正確使用方式,避免因?yàn)槭聞?wù)處理不當(dāng)導(dǎo)致業(yè)務(wù)邏輯產(chǎn)生大量偶發(fā)性BUG。

事務(wù)的傳播類(lèi)型

//如果沒(méi)有事務(wù)就進(jìn)行創(chuàng)建,存在則加入
@Transactional(propagation=Propagation.REQUIRED)

//不為當(dāng)前方法開(kāi)啟事務(wù) 
@Transactional(propagation=Propagation.NOT_SUPPORTED)

//不管是否存在事務(wù), 都創(chuàng)建一個(gè)新的事務(wù), 原來(lái)的掛起, 新的執(zhí)行完畢后, 繼續(xù)執(zhí)行老的事務(wù) 
@Transactional(propagation=Propagation.REQUIRES_NEW) 

//必須在一個(gè)已有的事務(wù)中執(zhí)行, 否則拋出異常
@Transactional(propagation=Propagation.MANDATORY) 

//必須在一個(gè)沒(méi)有的事務(wù)中執(zhí)行, 否則拋出異常(與Propagation.MANDATORY相反)
@Transactional(propagation=Propagation.NEVER) 

//如果其他bean調(diào)用這個(gè)方法, 在其他bean中聲明事務(wù), 那就用事務(wù), 如果其他bean沒(méi)有聲明事務(wù), 那就不用事務(wù)
@Transactional(propagation=Propagation.SUPPORTS) 

事務(wù)隔離級(jí)別

// 讀未提交(會(huì)出現(xiàn)臟讀, 不可重復(fù)讀) 基本不使用
@Transactional(isolation = Isolation.READ_UNCOMMITTED)

// 讀已提交(會(huì)出現(xiàn)不可重復(fù)讀和幻讀) Oracle默認(rèn)
@Transactional(isolation = Isolation.READ_COMMITTED)

// 可重復(fù)讀(會(huì)出現(xiàn)幻讀) MySQL默認(rèn)
@Transactional(isolation = Isolation.REPEATABLE_READ)

// 串行化
@Transactional(isolation = Isolation.SERIALIZABLE)

事務(wù)失效的場(chǎng)景

  • 事務(wù)方法未被Spring管理

如果事務(wù)方法所在的類(lèi)沒(méi)有注冊(cè)到Spring IOC容器中,也就是說(shuō),事務(wù)方法所在類(lèi)并沒(méi)有被Spring管理,則Spring事務(wù)會(huì)失效,舉個(gè)例子??:

public class BackGroupServiceImpl {
    @Autowired
    private SelfHelpBackgroundMapper backgroundMapper;
    @Transactional(propagation = Propagation.REQUIRES_NEW)
    public void updateSelfHelpBackground(SelfHelpBackground background) {
        backgroundMapper.updateByPrimaryKey(background);
    }
}

BackGroupServiceImpl 實(shí)現(xiàn)類(lèi)上沒(méi)有添加 @Service注解,實(shí)例也就沒(méi)有被加載到Spring IOC容器,此時(shí)updateSelfHelpBackground()方法的事務(wù)就會(huì)在Spring中失效。

  • 同一個(gè)類(lèi)中的事務(wù)方法被非事務(wù)方法調(diào)用
@Service
public class BackGroupServiceImpl {
    @Autowired
    private SelfHelpBackgroundMapper backgroundMapper;
    @Transactional(propagation = Propagation.REQUIRED)
    public  void updateSelfHelpBackground(SelfHelpBackground background) {
        backgroundMapper.updateByPrimaryKey(background);
    }
    public void updateBackground(){
        updateSelfHelpBackground(new SelfHelpBackground());
    }
}

updateBackgroup()方法和updateSelfHelpBackgroup()方法都在BackGroupServiceImpl類(lèi)中,然而updateBackgroup()方法沒(méi)有添加事務(wù)注解,updateSelfHelpBackgroup()方法雖然添加了事務(wù)注解,這種情況updateSelfHelpBackgroup()會(huì)在Spring事務(wù)中失效。

  • 方法的事務(wù)傳播類(lèi)型不支持事務(wù)
@Service
public class BackGroupServiceImpl {
    @Autowired
    private SelfHelpBackgroundMapper backgroundMapper;
    @Transactional(propagation = Propagation.REQUIRED)
    public  void updateSelfHelpBackground(SelfHelpBackground background) {
        backgroundMapper.updateByPrimaryKey(background);
    }
    @Transactional(propagation = Propagation.NOT_SUPPORTED)
    public void updateBackground(SelfHelpBackground background){
       backgroundMapper.updateByPrimaryKey(background);
    }
}

如果內(nèi)部方法的事務(wù)傳播類(lèi)型為不支持事務(wù)的傳播類(lèi)型,則內(nèi)部方法的事務(wù)同樣會(huì)在Spring中失效,如@Transactional(propagation = Propagation.NOT_SUPPORTED)

  • 異常被內(nèi)部catch,程序生吞異常
@Service
public class OrderServiceImpl{
    @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);
        // 扣減庫(kù)存
        this.updateProductStockById(order.getProductId(), 1L);
        return new ResponseEntity(HttpStatus.OK);
    }
    /**
     * 事務(wù)類(lèi)型聲明為NOT_SUPPORTED不支持事務(wù)的傳播
     */
    @Transactional(propagation = Propagation.NOT_SUPPORTED)
    public void updateProductStockById(Integer num, Long productId) {
        try {
            productMapper.updateProductStockById(num, productId);
        } catch (Exception e) {
            // 這里僅僅是捕獲異常之后的打?。ㄏ喈?dāng)于程序吞掉了異常)
            log.error("Error updating product Stock: {}", e);
        }
    }
}
  • 數(shù)據(jù)庫(kù)不支持事務(wù)

Spring事務(wù)生效的前提是連接的數(shù)據(jù)庫(kù)支持事務(wù),如果底層的數(shù)據(jù)庫(kù)都不支持事務(wù),則Spring事務(wù)肯定會(huì)失效的,例如??:使用MySQL數(shù)據(jù)庫(kù),選用MyISAM存儲(chǔ)引擎,因?yàn)?code>MyISAM存儲(chǔ)引擎本身不支持事務(wù),因此事務(wù)毫無(wú)疑問(wèn)會(huì)失效

  • 未配置開(kāi)啟事務(wù)

如果項(xiàng)目中沒(méi)有配置Spring的事務(wù)管理器,即使使用了Spring的事務(wù)管理功能,Spring的事務(wù)也不會(huì)生效,例如,如果你是Spring Boot項(xiàng)目,沒(méi)有在SpringBoot項(xiàng)目中配置如下代碼:

@Bean
public PlatformTransactionManager transactionManager(DataSource dataSource) {
    return new DataSourceTransactionManager(dataSource);
}
  • 多線(xiàn)程調(diào)用
@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ā)送短信
    }
}

通過(guò)示例,我們可以看到訂單提交的事務(wù)方法orderCommit()中,調(diào)用了發(fā)送短信的事務(wù)方法sendSms(),但是發(fā)送短信的事務(wù)方法sendSms()是另起了一個(gè)線(xiàn)程調(diào)用的。

這樣會(huì)導(dǎo)致兩個(gè)方法不在同一個(gè)線(xiàn)程中,從而是兩個(gè)不同的事務(wù)。如果是sendSms()方法中拋了異常,orderCommit()方法也回滾是不可能的。

實(shí)際上,Spring的事務(wù)是通過(guò)ThreadLocal來(lái)保證線(xiàn)程安全的,事務(wù)和當(dāng)前線(xiàn)程綁定,多個(gè)線(xiàn)程自然會(huì)讓事務(wù)失效。

到此這篇關(guān)于Spring事務(wù)失效的場(chǎng)景梳理總結(jié)的文章就介紹到這了,更多相關(guān)Spring事務(wù)失效場(chǎng)景內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • SpringBoot+SpringSecurity處理Ajax登錄請(qǐng)求問(wèn)題(推薦)

    SpringBoot+SpringSecurity處理Ajax登錄請(qǐng)求問(wèn)題(推薦)

    這篇文章主要介紹了SpringBoot+SpringSecurity處理Ajax登錄請(qǐng)求問(wèn)題,本文給大家介紹的非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下
    2017-12-12
  • 編程入門(mén):掌握J(rèn)ava運(yùn)算符技巧

    編程入門(mén):掌握J(rèn)ava運(yùn)算符技巧

    掌握J(rèn)ava運(yùn)算符技巧,能讓你的編程之旅輕松許多,本指南將帶你深入了解如何巧妙地使用這些強(qiáng)大的工具,讓代碼不僅高效,還充滿(mǎn)樂(lè)趣,跟著我們一起,讓你的Java代碼在運(yùn)算符的魔法下煥發(fā)新生!
    2023-12-12
  • Spring Boot支持Crontab任務(wù)改造的方法

    Spring Boot支持Crontab任務(wù)改造的方法

    這篇文章主要介紹了Spring Boot支持Crontab任務(wù)改造的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2019-01-01
  • 教你用Java Swing做一個(gè)定時(shí)提醒工具

    教你用Java Swing做一個(gè)定時(shí)提醒工具

    今天給大家?guī)?lái)的是Java的相關(guān)知識(shí),文章圍繞著如何用Java做一個(gè)定時(shí)提醒工具展開(kāi),文中有非常詳細(xì)的介紹及代碼示例,需要的朋友可以參考下
    2021-06-06
  • springBoot整合RocketMQ及坑的示例代碼

    springBoot整合RocketMQ及坑的示例代碼

    這篇文章主要介紹了springBoot整合RocketMQ及坑的示例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-11-11
  • Java中Random簡(jiǎn)介_(kāi)動(dòng)力節(jié)點(diǎn)Java學(xué)院整理

    Java中Random簡(jiǎn)介_(kāi)動(dòng)力節(jié)點(diǎn)Java學(xué)院整理

    本文詳細(xì)給大家介紹了Java中Random簡(jiǎn)介相關(guān)知識(shí),非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友參考下吧
    2017-06-06
  • idea插件生成jpa實(shí)體類(lèi)的實(shí)現(xiàn)示例

    idea插件生成jpa實(shí)體類(lèi)的實(shí)現(xiàn)示例

    本文主要介紹了idea插件生成jpa實(shí)體類(lèi)的實(shí)現(xiàn)示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2024-01-01
  • IDEA修改生成jar包名字的兩種方法實(shí)現(xiàn)

    IDEA修改生成jar包名字的兩種方法實(shí)現(xiàn)

    本文主要介紹了IDEA修改生成jar包名字的兩種方法實(shí)現(xiàn),通過(guò)簡(jiǎn)單的步驟,您可以修改項(xiàng)目名稱(chēng)并在打包時(shí)使用新的名稱(chēng),具有一定的參考價(jià)值,感興趣的可以了解下
    2023-08-08
  • Java線(xiàn)程阻塞和喚醒的幾種方式詳解

    Java線(xiàn)程阻塞和喚醒的幾種方式詳解

    這篇文章主要介紹了Java線(xiàn)程阻塞和喚醒的幾種方式詳解,線(xiàn)程阻塞是指當(dāng)一個(gè)線(xiàn)程無(wú)法繼續(xù)執(zhí)行時(shí),它會(huì)進(jìn)入阻塞狀態(tài),直到某個(gè)條件滿(mǎn)足后才能繼續(xù)執(zhí)行,線(xiàn)程阻塞可以通過(guò)多種方式實(shí)現(xiàn),如等待鎖、等待IO操作、等待其他線(xiàn)程的完成等,需要的朋友可以參考下
    2023-10-10
  • Spring?MVC策略模式之MethodArgumentResolver源碼解析

    Spring?MVC策略模式之MethodArgumentResolver源碼解析

    這篇文章主要為大家介紹了Spring?MVC策略模式之MethodArgumentResolver源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-03-03

最新評(píng)論