事务的传播行为

举例:

用户想要买两本书,但是账户的钱只够支付一本书,这时需要先支付一本。
如果事默认的事务传播行为(REQUIRED),则两本书都不能购买成功,这是需要将事务的传播行为设置为REQUIRES_NEW
重点事这两种传播行为:
REQUIRED 传播行为:
默认的事务传播行为,当前方法内的所有方法都在一个事务中,要么都成功,要么都不成功。

@Transactional(propagation=Propagation.REQUIRED)

REQUIRED_NEW传播行为:
它表示该方法必须启动一个新事务, 并在自己的事务内运行. 如果有事务在运行, 就应该先挂起它.

@Transactional(propagation=Propagation.REQUIRES_NEW)

事务传播行为在配置文件中配置:

<tx:advice id="txAdvice" transaction-manager="transactionManager">
        <tx:attributes>
            <tx:method name="*" propagation="REQUIRES_NEW"/>
        </tx:attributes>
    </tx:advice>

下面放上具体的例子:
模拟用户买书,如果一个用户想要买两本书,但是账户余额不足时,可以让用户先买一本

package com.nxmu.spring.TransactionManager;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Repository;

@Repository("bookShopDao")
public class BookShopDaoImpl implements BookShopDao{

    @Autowired
    private JdbcTemplate jdbcTemplate;

    /**
     * 查询书的价格
     */
    @Override
    public int findBookPriceByIsbn(String isbn) {
        String sql = "select price from book where isbn = ?";
        int  price = jdbcTemplate.queryForObject(sql, Integer.class,isbn);
        return price;
    }

    /**
     * 更新书的数量
     */
    @Override
    public void updateBookStock(String isbn) {
        String sql2 = "select stock from book_stock where isbn = ?";
        int stock = jdbcTemplate.queryForObject(sql2, Integer.class,isbn);
        if (stock == 0) {
            throw new BookStockException("库存不足");
        }

        String sql = "update book_stock set stock = stock - 1 where isbn = ?";
        jdbcTemplate.update(sql,isbn);
    }

    /**
     * 更新账户余额
     */
    @Override
    public void updateUserAccount(String username, int price) {

        String sql2 = "select balance from account where username = ?";
        int balance = jdbcTemplate.queryForObject(sql2, Integer.class,username);
        if (balance < price) {
            throw new UserAccountException("余额不足");
        }
        String sql = "update account set balance = balance - ? where username = ?";
        jdbcTemplate.update(sql,price,username);
    }

}

这里将购买每本书的方法事务传播行为设置为REQUIRED_NEW

@Service("bookShopService")
public class BookShopServiceImpl implements BookShopService {

    @Autowired
    private BookShopDao bookShopDao;

    @Transactional(propagation=Propagation.REQUIRES_NEW)
    @Override
    public void purchese(String username, String isbn) {

        int price = bookShopDao.findBookPriceByIsbn(isbn);

        bookShopDao.updateBookStock(isbn);

        bookShopDao.updateUserAccount(username, price);

    }

}

下面的类时用来一次购买多本书,传入用户名和需要购买书的id的list,对list进行遍历,每一次便利就是购买一本书,每一次购买书就时一次事务

@Service("cashier")
public class CachierImpl implements Cashier {

    @Autowired
    private BookShopService bookShopService;

    @Transactional(propagation=Propagation.REQUIRED)
    @Override
    public void checkout(String username, List<String> isbns) {
        for (String isbn : isbns) {
            bookShopService.purchese(username, isbn);
        }
    }

}

下面时测试方法:

    @Test
    void testCheckout() {
        List<String> isbns = new ArrayList<>();
        cashier.checkout("AA", Arrays.asList("1001","1002"));
    }

这里写图片描述

### 事务传播行为概述 在Spring框架中,事务传播行为定义了一个方法调用链中多个事务方法之间如何协作。具体来说,这决定了当一个事务方法被另一个事务方法调用时,两个方法间的事务边界和一致性应如何处理[^3]。 #### 默认传播行为REQUIRED `REQUIRED`是Spring框架中的默认事务传播行为。这意味着每当一个带有此属性的方法被执行: - 如果当前已经存在一个活动的事务,则新方法将加入这个现有的事务; - 若无现有事务,则会启动一个新的事务来执行该操作[^2]。 这种方法适用于大多数情况下的业务需求,因为它确保了所有必要的数据库交互都在同一个事务上下文中完成,从而保持数据的一致性和完整性。 #### 特殊传播行为:NEVER 对于某些特定的操作,可能希望它们始终不在任何事务内运行。例如,在日志记录或其他不需要回滚机制的服务里。这时可以采用`Propagation.NEVER`策略。这种情况下,如果尝试在一个已有事务环境中调用此类服务,将会触发异常并阻止进一步的动作[^5]。 ```java @Service @Transactional public class OrderServiceImpl implements OrderService { @Autowired private OrderDao orderDao; @Override public void createOrder(Order order) { // 创建订单的具体业务逻辑 } @Override @Transactional(propagation = Propagation.NEVER) public void deleteOrder(int orderId) { Order order = orderDao.getOrderById(orderId); orderDao.deleteOrder(order); } } ``` 上述代码展示了如何通过设置不同的传播行为来控制不同类型的业务流程。这里`createOrder()`遵循默认的`REQUIRED`模式,而`deleteOrder()`则强制要求不处于任何事务之中。 ### 使用场景分析 1. **跨库操作**:当应用程序涉及多数据库访问时,合理的配置事务传播行为可以帮助解决分布式系统的复杂性问题。 2. **异步任务处理**:对于那些不应该参与主线程事务的任务(比如发送邮件),可以选择合适的传播级别以避免不必要的锁定或资源占用。 3. **读取密集型应用**:为了提高性能,可以在只做查询的地方禁用事务支持,减少开销。 ### 示例总结 综上所述,理解并正确运用Spring中的事务传播行为能够极大地增强程序设计灵活性以及应对各种实际应用场景的能力。合理选择适合项目特点的行为类型不仅有助于简化开发过程,还能有效提升系统的稳定性和效率。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值