spring自己没有事务,spring事务是在数据库事务的基础上进行封装拓展,spring支持声明式事务、编程式事务两种,本文主要针对声明式事务进行讲解,本篇文章为《图灵学院》课程笔记java
咱们在使用事务的时候无非是如下几步git
其实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事务特性以下
属性 | 属性值 | 描述 |
---|---|---|
支持当前事物 | PROPAGATION_REQUIRED(必须的) | 若是当前没有事物,就新建一个事物,若是已经存在一个事物中,加入到这个事物中。这是最多见的选择。 |
PROPAGATION_SUPPORTS(支持) | 支持当前事物,若是当前没有事物,就以非事物方式执行。 | |
PROPAGATION_MANDATORY(强制) | 使用当前的事物,若是当前没有事物,就抛出异常。 | |
不支持当前事物 | PROPAGATION_REQUIRES_NEW(隔离) | 新建事物,若是当前存在事物,把当前事物挂起。 |
PROPAGATION_NOT_SUPPORTED(不支持) | 以非事物方式执行操做,若是当前存在事物,就把当前事物挂起。 | |
PROPAGATION_NEVER(强制非事物) | 以非事物方式执行,若是当前存在事物,则抛出异常。 | |
套事物 | PROPAGATION_NESTED(嵌套事物) | 若是当前存在事物,则在嵌套事物内执行。若是当前没有事物,则执行与PROPAGATION_REQUIRED相似的操做。 |
隔离级别 | 脏读(Dirty Read) | 不可重复读(NonRepeatable Read) | 幻读(Phantom Read) |
---|---|---|---|
未提交读(Read uncommitted) | 可能 | 可能 | 可能 |
已提交读(Read committed) | 不可能 | 可能 | 可能 |
可重复读(Repeatable read) | 不可能 | 不可能 | 可能 |
可串行化(SERIALIZABLE) | 不可能 | 不可能 | 不可能 |
脏读 :
一个事物读取到另外一事物未提交的更新数据
不可重复读 :
在同一事物中,屡次读取同一数据返回的结果有所不一样, 换句话说, 后续读取能够读到另外一事物已提交的更新数据. 相反, “可重复读”在同一事物中屡次读取数据时, 可以保证所读数据同样, 也就是后续读取不能读到另外一事物已提交的更新数据。
幻读 :
查询表中一条数据若是不存在就插入一条,并发的时候却发现,里面竟然有两条相同的数据。这就幻读的问题。
spring提供三个接口提供使用事务:
TransactionDefinition 事务定义
PlatformTransactionManager 事务管理
TransactionStatus 事务运行时状态
原文地址