SpringBoot 的事务及使用

一、事务的常识

1、事务四特性(ACID)

A 原子性:事务是最小单元,不可再分隔的一个整体。
C 一致性:事务中的方法要么同时成功,要么都不成功,要不都失败。
I 隔离性:多个事务操作数据库中同一个记录或多个记录时,对事务进行隔离开来有序执行。
D 持久性:事务成功时,操作的结果永久的写入到数据库磁盘中。

2、五种数据库BUG读

1.脏读:(完全不加锁,且直接改主数据数据)
   A线程写时不加锁,B线程读A线程未提交的数据。A回滚,B线程之前读到的数据为无效数据。
2.不可重复读:(读不加共享锁)
   A事务第一次读数据,B线程改操作。A第二次读取数据,同一事物内两次读取数据不一致。
3.第一类事务丢失:(回滚丢失)
   A事务撤销时,在A事务开始和结束的B事务也抹杀了,无视B的存在。
4.第二类事务丢失:(提交覆盖丢失)
   A事务覆盖B事务已提交的数据,造成B事务操作丢失。
5.幻读:(不锁表)
   主要指两次读取表的总数不一致。

3、四种隔离级别(就是共享锁、独占锁、表锁)

1.读未提交数据(read uncommitted)(完全不加锁,且直接改主数据数据)
    允许事务读取未被其他事务提交的变更。
2.读已提交数据(read committed)(写时加独占锁,读时不加锁)
    只允许事务读取已经被其他事务提交的变更。
3.可重复读(repeatable read)(共享锁、与独占锁,默认)
    确保事务可以多次从一个字段中读取相同的值,在这个事务持续期间,禁止其他事务对这个字段进行更新。
4.串行化(serializable)(对表级加锁)
    读操作加表级读锁至事务结束。可以禁止幻读。会导致大量的操作超时和锁竞争,大大降低数据库的性。

4、Spring事务的7种传播行为

定义多个事务方法之间如何共享事务上下文,
在Spring框架中,一个事务方法可以调用另一个事务方法,这时就需要考虑事务的传播机制。
如果调用的方法已经有了一个事务,那么新的方法可以加入该事务;如果调用的方法没有事务,那么新的方法可以开启一个新的事务。

(required / supports / mandatory / requires_new / not supported / never / nested)
required(加入):Spring默认的,加入当前事务,没有事务就新建。
supports(加入):有则加入,没有就不管了,非事务运行
mandatory(强制事务执行):有则加入,没有就抛异常。
requires_new(新建):不管有没有,直接创建新事务。
not supported(不支持):以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
never(不支持):以非事务方式执行,如果当前存在事务,则抛出异常。
nested(存在就嵌套的执行):当前存在事务,则嵌套在当前事务中执行。当前没事务,则新建,类似 REQUIRE_NEW。
对事务的要求程度可以从大到小排序:mandatory / supports / required / requires_new / nested / not supported / never;

二、编程式事务管理

用 TransactionTemplate 或 PlatformTransactionManager,Spring 推荐 TransactionTemplate。

1、引入依赖

<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-jdbc</artifactId>
    <version>5.2.9.RELEASE</version>
</dependency>

2、代码

@Autowired
private PlatformTransactionManager transactionManager;

@Autowired
private TransactionTemplate transactionTemplate;

public void testTransactionManager() {
    TransactionStatus status = transactionManager.getTransaction(new DefaultTransactionDefinition());
    try {
        // ....  业务代码
        transactionManager.commit(status);
    } catch (Exception e) {
        transactionManager.rollback(status);
    }
}

public void testTransactionTemplate() {
    transactionTemplate.execute(new TransactionCallbackWithoutResult() {
        @Override
        protected void doInTransactionWithoutResult(TransactionStatus transactionStatus) {
            try {
                // ....  业务代码
            } catch (Exception e) {
                //回滚
                transactionStatus.setRollbackOnly();
            }
        }
    });
}

三、声明式事务(@Transactional)

Spring 用 ThreadLocal 为线程做副本,存放维护事务配置的属性、运行状态等信息,
通过 AOP 机制在方法前后分别织入开启事务、提交、回滚的逻辑,
@Transactional 注解修饰的方法前后分别织入开启事务的逻辑,以及提交或回滚的逻辑。
@Transactional 可以修饰在方法或者类上,区别就在于修饰于类上的,会对该类下符合条件的方法(例如private修饰的方法就不符合条件)前后都织入事务的逻辑。

失 效 场 景:
1.protected、private 修饰的方法上使用 @Transactional 注解,虽然事务无效,但不会有任何报错。
2.类内方法调用会失效。
3.和线程池使用。

1、引入依赖

<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-jdbc</artifactId>
    <version>5.2.9.RELEASE</version>
</dependency>

2、代码

@Transactional(propagation= Propagation.REQUIRED,isolation =Isolation.READ_UNCOMMITTED,timeout=30,readOnly=true,rollbackFor=RuntimeException.class)
public void testTransaction(){
        // ....  业务代码
}
  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值