当前位置: 首页 > news >正文

Spring学习笔记_29——@Transactional

@Transactional

1. 介绍

@Transactional 是 Spring 框架提供的一个注解,用于声明方法或类级别的事务属性。

Spring事务:Spring学习笔记_28——事务-CSDN博客

当你在一个方法或类上使用 @Transactional 注解时,Spring 会为该方法或类创建一个事务代理,以确保方法的执行是在事务的上下文中进行的。

2. 场景

@Transactional注解在 Spring 框架中被广泛用于需要事务管理的业务场景。以下是一些常见的业务场景,其中@Transactional注解的使用是非常合适的:

  • 数据一致性保证
    • 在涉及多个数据库操作,需要保证这些操作要么全部成功,要么全部失败的场景中,使用 @Transactional 可以确保数据的一致性。
  • 并发事务处理
    • 当多个用户或服务同时对同一数据进行操作时,使用 @Transactional 可以防止数据竞争条件和不一致性。
  • 资金交易
    • 在电子商务或金融应用中,资金转账、支付处理等场景需要精确的事务控制,以确保资金的正确流动。
  • 库存管理
    • 在订单处理系统中,库存的增减需要在事务控制下进行,以避免超卖或库存不一致的问题。
  • 用户注册和登录
    • 用户注册时创建用户记录和相关配置信息,或者登录时更新用户状态,这些操作需要在事务控制下确保数据的一致性。
  • 数据批量处理
    • 在处理批量数据导入或导出时,需要确保所有数据项都成功处理,或者在遇到错误时回滚,以保持数据的完整性。
  • 复合操作
    • 当需要执行一系列复合操作,如更新多个相关联的表或记录时,使用 @Transactional 可以确保这些操作作为一个整体被处理。
  • 错误恢复
    • 在业务操作中,如果遇到异常需要回滚到操作前的状态,@Transactional 可以自动处理这种回滚。
  • 缓存与数据库同步
    • 当更新数据库的同时需要更新缓存,以保持缓存数据的一致性,@Transactional 可以确保数据库和缓存的更新操作要么同时成功,要么同时失败。
  • 分布式事务
    • 在微服务架构中,跨服务的数据操作可能需要分布式事务来保证数据的一致性。
  • 异步消息处理
    • 在处理消息队列中的消息时,可能需要确保消息的发送和数据库操作的一致性。
  • 服务层方法
    • 在服务层封装业务逻辑时,如果业务逻辑涉及到多个数据访问操作,使用 @Transactional 可以简化事务管理。
  • 补偿事务
    • 在需要执行补偿操作的场景中,如果一个操作失败,需要执行另一个操作来补偿,@Transactional 可以确保这两个操作的一致性。

3. 源码

/*** @author Colin Sampaleanu* @author Juergen Hoeller* @author Sam Brannen* @author Mark Paluch* @since 1.2* @see org.springframework.transaction.interceptor.TransactionAttribute* @see org.springframework.transaction.interceptor.DefaultTransactionAttribute* @see org.springframework.transaction.interceptor.RuleBasedTransactionAttribute*/
@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
@Reflective
public @interface Transactional {// String类型,用以指定事务管理器的唯一标识@AliasFor("transactionManager")String value() default "";/*** @since 4.2*/// 用以指定事务管理器的唯一标识@AliasFor("value")String transactionManager() default "";/*** @since 5.3*/// 设置属性的标签String[] label() default {};// Propagation枚举类型的属性,指定事务的隔离级别Propagation propagation() default Propagation.REQUIRED;// isolation枚举类型的属性,指定事务隔离的级别,Isolation isolation() default Isolation.DEFAULT;// timeout指定事务的超时时间,单位为秒,当事务执行时间超过timeout秒时,就会触发超时回滚操作,并释放事务占用的资源int timeout() default TransactionDefinition.TIMEOUT_DEFAULT;/*** @since 5.3*/// 以String类型设置超时时间,单位为秒String timeoutString() default "";// 指定是否为只读事务// ture:只读// false:非只读boolean readOnly() default false;// 指定异常类的Class对象,当抛出指定类型异常或者其子类型的异常时,事务会自动回滚Class<? extends Throwable>[] rollbackFor() default {};// 指定异常类的全类名,当抛出指定全类名的异常或者其子类型的异常时,事务自动回滚String[] rollbackForClassName() default {};// 指定异常类的class对象,当抛出指定类型的异常或者其子类型的异常时,事务不会自动回滚Class<? extends Throwable>[] noRollbackFor() default {};// 指定异常类的全类名,当抛出指定全类名的异常或者其子类型的异常时,事务不会自动回滚String[] noRollbackForClassName() default {};
}

4. Demo

  • 不同传播行为
@Transactional(propagation = Propagation.REQUIRED)
public void updateProductStock(Long productId, int quantity) {// 业务逻辑:更新产品库存
}
  • 不同隔离级别
@Transactional(isolation = Isolation.READ_COMMITTED)
public void processOrder(Long orderId) {// 以 READ_COMMITTED 隔离级别执行的业务逻辑
}
  • 超时时间
@Transactional(timeout = 5) // 超时时间设置为5秒
public void longRunningProcess() {// 长时间运行的业务逻辑
}
  • 只读事务
@Transactional(readOnly = true)
public List<Product> getAllProducts() {// 只读事务,用于查询所有产品的列表
}
  • 设置回滚异常
@Transactional(rollbackFor = Exception.class) // 默认所有异常都会触发回滚
public void processWithCustomRollback(Long productId) {// 如果抛出Exception类型的异常,事务将回滚
}@Transactional(noRollbackFor = ArithmeticException.class)
public void processWithNoRollback(Long productId) {// 如果抛出ArithmeticException类型的异常,事务不会回滚
}
  • 组合应用
@Transactional(propagation = Propagation.REQUIRES_NEW, isolation = Isolation.SERIALIZABLE, timeout = 10, readOnly = false, rollbackFor = Exception.class)
public void complexBusinessProcess(Long productId, int quantity) {// 组合使用不同的@Transactional参数// 这个操作将在一个新的事务中执行,具有SERIALIZABLE隔离级别,超时时间为10秒,非只读事务,并且所有异常都会触发回滚
}

http://www.lryc.cn/news/480298.html

相关文章:

  • github使用基础
  • Flink-Kafka-Connector
  • 远程终端vim里使用系统剪切板
  • 底层视角看C语言
  • 【点云学习笔记】——分割任务学习
  • Qt——窗口
  • InfluxDB性能优化指南
  • 负载均衡式在线oj项目开发文档2(个人项目)
  • ssm081高校实验室管理系统的设计与实现+vue(论文+源码)_kaic
  • GitLab基于Drone搭建持续集成(CI/CD)
  • 用GPB外链打造长期稳定的SEO优势!
  • 第11章 内连接与外连接
  • C++ 游戏开发:打造高效、性能优越的游戏世界
  • 太速科技-440-基于XCVU440的多核处理器多输入芯片验证板卡
  • 澳鹏通过高质量数据支持 Onfido 优化AI反欺诈功能
  • 基于ECS实例搭建Hadoop环境
  • 关于vue如何监听route和state以及各自对应的实际场景
  • 【计网不挂科】计算机网络期末考试(综合)——【选择题&填空题&判断题&简述题】完整题库
  • Linux(CentOS)设置防火墙开放8080端口,运行jar包,接收请求
  • 对比:生成对抗网络(GANs)和变分自编码器(VAEs)
  • sqlserver inner join on 条件是包含 怎么写
  • 开源 AI 智能名片 S2B2C 商城小程序在微商内容展示中的应用与价值
  • Codeforces Round 984 (Div. 3) (A~E)
  • pytorch3d报错:RuntimeError: Not compiled with GPU support.
  • 软考中级-软件设计师 数据结构与算法
  • 关于CSS表达使中使用的 max() 函数
  • 51单片机教程(八)- 数码管的静态显示
  • 案例精选 | 河北省某检察院安全运营中异构日志数据融合的实践探索
  • clickhouse自增id的处理
  • 国内读新加坡公立大学在职博士是一种怎样的体验?还中文授课