spring|事务底层原理分析

spring自己没有事务,spring事务是在数据库事务的基础上进行封装拓展,spring支持声明式事务、编程式事务两种,本文主要针对声明式事务进行讲解,本篇文章为《图灵学院》课程笔记java

咱们在使用事务的时候无非是如下几步git

  • 获取数据库链接 Connection con = DriverManager.getConnection()
  • 开启事务con.setAutoCommit(true/false);
  • 执行数据操做(crud)
  • 提交事务/回滚事务 con.commit() / con.rollback()
  • 关闭链接 conn.close()

其实spring是在框架中给我咱们作了开启、提交/回滚的操做,使得业务代码和事务操做解耦。那么spring是如何实如今指定方法先后自动加上事务操做的呢;这就要扯到动态代理了spring

代理模式

  • 解决的问题:将次要业务、主要业务解耦。,次要业务:起到辅助做用,辅助主要业务顺利实现,在项目中每每大量存在数据库

具体例子以下:编程

接口markdown

public interface UserDao {
     void save();
}
复制代码

接口实现、目标对象并发

public class UserDaoImpl implements UserDao {
    @Override
    public void save() {
        System.out.println("数据已经保存");
    }
}
复制代码

代理对象框架

public class ProxyFactory {

    private  Object target;

    public ProxyFactory(Object object) {
        this.target = object;
    }

    public Object getProxyObject(){
       return  Proxy.newProxyInstance(target.getClass().getClassLoader(),
                 target.getClass().getInterfaces(),new InvocationHandler(){
                   /** * @param proxy 负责监听的对象 * @param method 被拦截的业务方法 * @param args 被拦截业务方法的实参 * @return * @throws Throwable */
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        System.out.println("before do something...");
                         Object returnValue= method.invoke(target,args);
                        System.out.println("after do something...");
                        return returnValue;
                    }
                });
    }
}
复制代码

测试方法ide

public static void main(String[] args) {
        //目标对象
        UserDao target = new UserDaoImpl();
        //目标对象建立代理对像
        UserDao proxy = (UserDao) new ProxyFactory(target).getProxyObject();
        proxy.save();
    }
复制代码

输出结果oop

结合着代理模式,spring事务的实现原理就容易理解了,业务逻辑就是主要业务,须要重复使用的事务就是次要业务,spring的事务就是经过动态代理在业务代码的先后增长开启、提交/回滚的操做,实现事务操做

spring事务特性以下

  • 支持原有数据事务的隔离级别
  • 加入事务传播特性概念,提供多个事务的合并或隔离的功能
  • 提供声明式事务,让业务代码和事务分离,事务简单易用

spring事务的传播特性

属性 属性值 描述
支持当前事物 PROPAGATION_REQUIRED(必须的) 若是当前没有事物,就新建一个事物,若是已经存在一个事物中,加入到这个事物中。这是最多见的选择。
PROPAGATION_SUPPORTS(支持) 支持当前事物,若是当前没有事物,就以非事物方式执行。
PROPAGATION_MANDATORY(强制) 使用当前的事物,若是当前没有事物,就抛出异常。
不支持当前事物 PROPAGATION_REQUIRES_NEW(隔离) 新建事物,若是当前存在事物,把当前事物挂起。
PROPAGATION_NOT_SUPPORTED(不支持) 以非事物方式执行操做,若是当前存在事物,就把当前事物挂起。
PROPAGATION_NEVER(强制非事物) 以非事物方式执行,若是当前存在事物,则抛出异常。
套事物 PROPAGATION_NESTED(嵌套事物) 若是当前存在事物,则在嵌套事物内执行。若是当前没有事物,则执行与PROPAGATION_REQUIRED相似的操做。

spring事务隔离级别

隔离级别 脏读(Dirty Read) 不可重复读(NonRepeatable Read) 幻读(Phantom Read)
未提交读(Read uncommitted) 可能 可能 可能
已提交读(Read committed) 不可能 可能 可能
可重复读(Repeatable read) 不可能 不可能 可能
可串行化(SERIALIZABLE) 不可能 不可能 不可能

脏读 :

一个事物读取到另外一事物未提交的更新数据

不可重复读 :

在同一事物中,屡次读取同一数据返回的结果有所不一样, 换句话说, 后续读取能够读到另外一事物已提交的更新数据. 相反, “可重复读”在同一事物中屡次读取数据时, 可以保证所读数据同样, 也就是后续读取不能读到另外一事物已提交的更新数据。

幻读 :

查询表中一条数据若是不存在就插入一条,并发的时候却发现,里面竟然有两条相同的数据。这就幻读的问题。

spring提供三个接口提供使用事务:

  • TransactionDefinition 事务定义

  • PlatformTransactionManager 事务管理

  • TransactionStatus 事务运行时状态

原文地址

cbaj.gitee.io/blog/2020/0…

相关文章
相关标签/搜索