兰州seo安安网站建设,网络营销怎么做网站,大连网站建设酷网科技,网站点击文章目录 1、事务传播行为注意事务传播行为在不同类之间调用生效Propagation.REQUIRED(默认传播行为)Propagation.REQUIRES_NEWPropagation.NESTED 2、事务的隔离级别隔离级别设置 3、设置事务异常回滚3.1、默认情况3.2、设置回滚异常3.3、设置不回滚的异常 4、超时时间5、只读… 文章目录 1、事务传播行为注意事务传播行为在不同类之间调用生效Propagation.REQUIRED(默认传播行为)Propagation.REQUIRES_NEWPropagation.NESTED 2、事务的隔离级别隔离级别设置 3、设置事务异常回滚3.1、默认情况3.2、设置回滚异常3.3、设置不回滚的异常 4、超时时间5、只读 Transactional 注解开启事务其中注解的各种属性详解 1、事务传播行为
事务传播行为详解
事务传播行为是为了解决业务层方法之间互相调用的事务问题
当事务方法被另一个事务方法调用时必须指定事务应该如何传播。例如方法可能继续在现有事务中运行也可能开启一个新事务并在自己的事务中运行。
举个例子我们在 A 类的aMethod()方法中调用了 B 类的 bMethod() 方法。这个时候就涉及到业务层方法之间互相调用的事务问题。如果我们的 bMethod()如果发生异常需要回滚如何配置事务传播行为才能让 aMethod()也跟着回滚呢这个时候就需要事务传播行为的知识了如果你不知道的话一定要好好看一下
如下 StudentService中的 changeName 方法有运行时异常
Service
public class TopService {Autowiredprivate StudentService studentService;//测试 事务的传播行为Transactionalpublic void topService(){studentService.changeAge();studentService.changeName();}
}Service
public class StudentService {Autowiredprivate StudentDao studentDao;//事务的传播行为Transactionalpublic void changeAge(){studentDao.updateAgeById(998, 1);}//事务的传播行为Transactionalpublic void changeName(){studentDao.updateNameById(dasdas, 1);int i 1/0;}}在TransactionDefinition定义中包括了如下几个表示传播行为的常量
public interface TransactionDefinition {int PROPAGATION_REQUIRED 0;int PROPAGATION_SUPPORTS 1;int PROPAGATION_MANDATORY 2;int PROPAGATION_REQUIRES_NEW 3;int PROPAGATION_NOT_SUPPORTED 4;int PROPAGATION_NEVER 5;int PROPAGATION_NESTED 6;......
}不过为了方便使用Spring 相应地定义了一个枚举类Propagation
package org.springframework.transaction.annotation;import org.springframework.transaction.TransactionDefinition;public enum Propagation {REQUIRED(TransactionDefinition.PROPAGATION_REQUIRED),SUPPORTS(TransactionDefinition.PROPAGATION_SUPPORTS),MANDATORY(TransactionDefinition.PROPAGATION_MANDATORY),REQUIRES_NEW(TransactionDefinition.PROPAGATION_REQUIRES_NEW),NOT_SUPPORTED(TransactionDefinition.PROPAGATION_NOT_SUPPORTED),NEVER(TransactionDefinition.PROPAGATION_NEVER),NESTED(TransactionDefinition.PROPAGATION_NESTED);private final int value;Propagation(int value) {this.value value;}public int value() {return this.value;}}注意事务传播行为在不同类之间调用生效
在同一个类中对于Transactional注解的方法调用事务传播行为不会生效。
这是因为Spring框架中使用代理模式实现了事务机制在同一个类中的方法调用并不经过代理而是通过对象的方法调用因此Transactional注解的设置不会被代理捕获也就不会产生任何事务传播行为的效果。
Propagation.REQUIRED(默认传播行为)
如果父方法有事务就加入父方法事务如果没有就新建自己独立的事务
传播行为在子方法的Transactional中通过propagation 进行设置。
下面代码中是父方法有事务的情况propagation 设置为Propagation.REQUIRED在topService()中调用了studentService.changeAge()和studentService.changeName()因为事务传播行为为REQUIRED所以changeAge()和changeName()方法在同一个事务中。
此时changeName()发生运行时异常两个方法同时回滚 年龄和名字均不会被修改。
Service
public class TopService {Autowiredprivate StudentService studentService;//测试 事务的传播行为Transactionalpublic void topService(){studentService.changeAge();studentService.changeName();}
}Service
public class StudentService {Autowiredprivate StudentDao studentDao;//事务的传播行为Transactional(propagation Propagation.REQUIRED)public void changeAge(){studentDao.updateAgeById(998, 1);}//事务的传播行为Transactional(propagation Propagation.REQUIRED)public void changeName(){studentDao.updateNameById(dasdas, 1);int i 1/0;}}Propagation.REQUIRES_NEW
不管父方法是否有事务我都新建事务都是独立的子方法都是独立的事务
下面代码中propagation 设置为Propagation.REQUIRES_NEW在topService()中调用了studentService.changeAge()和studentService.changeName()因为事务传播行为为REQUIRES_NEW所以changeAge()和changeName() 子方法是两个独立的事务
此时changeName()发生运行时异常changeName()发生回滚不会影响changeAge()方法年龄将被修改名字不会修改。
Service
public class TopService {Autowiredprivate StudentService studentService;//测试 事务的传播行为Transactionalpublic void topService(){studentService.changeAge();studentService.changeName();}
}Service
public class StudentService {Autowiredprivate StudentDao studentDao;//事务的传播行为Transactional(propagation Propagation.REQUIRES_NEW)public void changeAge(){studentDao.updateAgeById(998, 1);}//事务的传播行为Transactional(propagation Propagation.REQUIRES_NEW)public void changeName(){studentDao.updateNameById(dasdas, 1);int i 1/0;}}Propagation.NESTED
如果当前存在事务就在嵌套事务内执行
在外围方法未开启事务的情况下Propagation.NESTED和Propagation.REQUIRED作用相同修饰的内部方法都会新开启自己的事务且开启的事务相互独立互不干扰在外围方法开启事务的情况下Propagation.NESTED修饰的内部方法属于外部事务的子事务外围主事务回滚子事务一定回滚而内部子事务可以单独回滚而不影响外围主事务和其他子事务
举个例子如果 bMethod() 回滚的话aMethod()不会回滚。如果 aMethod() 回滚的话bMethod()会回滚。
Service
Class A {AutowiredB b;Transactional(propagation Propagation.REQUIRED)public void aMethod {//do somethingb.bMethod();}
}Service
Class B {Transactional(propagation Propagation.NESTED)public void bMethod {//do something}
}
2、事务的隔离级别
隔离级别在Transactional中通过 isolation Isolation.XXXX 进行设置。
数据库事务的隔离级别是指在多个事务并发执行时数据库系统为了保证数据一致性所遵循的规定。常见的隔离级别包括
读未提交READ_UNCOMMITTED事务可以读取未被提交的数据容易产生脏读、不可重复读和幻读等问题。实现简单但不太安全一般不用。读已提交READ_COMMITTED事务只能读取已经提交的数据可以避免脏读问题但可能引发不可重复读和幻读。可重复读REPEATABLE_READ在一个事务中相同的查询将返回相同的结果集不管其他事务对数据做了什么修改。可以避免脏读和不可重复读但仍有幻读的问题。串行化SERIALIZABLE最高的隔离级别完全禁止了并发只允许一个事务执行完毕之后才能执行另一个事务。可以避免以上所有问题但效率较低不适用于高并发场景。
关于四种事务隔离级别和什么是脏读、不可重复读和幻读具体可查看MYSQL事务隔离级别知识点。
TransactionDefinition 接口中定义了五个表示隔离级别的常量
public interface TransactionDefinition {......int ISOLATION_DEFAULT -1;int ISOLATION_READ_UNCOMMITTED 1;int ISOLATION_READ_COMMITTED 2;int ISOLATION_REPEATABLE_READ 4;int ISOLATION_SERIALIZABLE 8;......
}
和事务传播行为那块一样为了方便使用Spring 也相应地定义了一个枚举类Isolation
public enum Isolation {DEFAULT(TransactionDefinition.ISOLATION_DEFAULT),READ_UNCOMMITTED(TransactionDefinition.ISOLATION_READ_UNCOMMITTED),READ_COMMITTED(TransactionDefinition.ISOLATION_READ_COMMITTED),REPEATABLE_READ(TransactionDefinition.ISOLATION_REPEATABLE_READ),SERIALIZABLE(TransactionDefinition.ISOLATION_SERIALIZABLE);private final int value;Isolation(int value) {this.value value;}public int value() {return this.value;}}隔离级别设置
隔离级别在Transactional中通过 isolation Isolation.XXXX 进行设置。
//isolation 设置事务的隔离级别,mysql默认是repeatable read! Transactional(isolation Isolation.REPEATABLE_READ)
public void changeInfo() throws FileNotFoundException {studentDao.updateAgeById(100,1);studentDao.updateNameById(test1,1);
}3、设置事务异常回滚
事务异常回滚在Transactional中通过 rollbackFor xxxException.class 进行设置。
3.1、默认情况
默认只针对运行时异常回滚编译时异常不回滚。情景模拟代码如下
下面程序会终止且不会回滚且 updateAgeById(100,1) 的方法 会修改数据库成功
Service
public class StudentService {Autowiredprivate StudentDao studentDao;/*** timeout设置事务超时时间,单位秒! 默认: -1 永不超时,不限制事务时间!* rollbackFor 指定哪些异常才会回滚,默认是 RuntimeException and Error 异常方可回滚!* noRollbackFor 指定哪些异常不会回滚, 默认没有指定,如果指定,应该在rollbackFor的范围内!*/Transactional(readOnly false,timeout 3)public void changeInfo() throws FileNotFoundException {studentDao.updateAgeById(100,1);//主动抛出一个检查异常,测试! 发现不会回滚,因为不在rollbackFor的默认范围内! new FileInputStream(xxxx);studentDao.updateNameById(test1,1);}
}3.2、设置回滚异常
rollbackFor属性指定哪些异常类才会回滚,默认是 RuntimeException and Error 异常方可回滚!
下面程序会终止但是会回滚因为 FileNotFoundException 属于 Exception异常updateAgeById方法修改数据库不成功。
/*** timeout设置事务超时时间,单位秒! 默认: -1 永不超时,不限制事务时间!* rollbackFor 指定哪些异常才会回滚,默认是 RuntimeException and Error 异常方可回滚!* noRollbackFor 指定哪些异常不会回滚, 默认没有指定,如果指定,应该在rollbackFor的范围内!*/
Transactional(readOnly false,timeout 3,rollbackFor Exception.class)
public void changeInfo() throws FileNotFoundException {studentDao.updateAgeById(100,1);//主动抛出一个检查异常,测试! 发现会回滚new FileInputStream(xxxx);studentDao.updateNameById(test1,1);
}3.3、设置不回滚的异常
在默认设置和已有设置的基础上再指定一个异常类型碰到它不回滚。
noRollbackFor属性指定哪些异常不会回滚, 默认没有指定,如果指定,应该在rollbackFor的范围内!
Service
public class StudentService {Autowiredprivate StudentDao studentDao;/*** timeout设置事务超时时间,单位秒! 默认: -1 永不超时,不限制事务时间!* rollbackFor 指定哪些异常才会回滚,默认是 RuntimeException and Error 异常方可回滚!* noRollbackFor 指定哪些异常不会回滚, 默认没有指定,如果指定,应该在rollbackFor的范围内!*/Transactional(readOnly false,timeout 3,rollbackFor Exception.class,noRollbackFor FileNotFoundException.class)public void changeInfo() throws FileNotFoundException {studentDao.updateAgeById(100,1);//主动抛出一个检查异常,测试! 发现不会回滚,因为设置了noRollbackFornew FileInputStream(xxxx);studentDao.updateNameById(test1,1);}
}4、超时时间
超时时间在Transactional中通过 timeout 3 进行设置。
事务在执行过程中有可能因为遇到某些问题导致程序卡住从而长时间占用数据库资源。而长时间占用资源大概率是因为程序运行出现了问题可能是Java程序或MySQL数据库或网络连接等等。
此时这个很可能出问题的程序应该被回滚撤销它已做的操作事务结束把资源让出来让其他正常程序可以执行。
概括来说就是一句话超时回滚释放资源。
Service
public class StudentService {Autowiredprivate StudentDao studentDao;/*** timeout设置事务超时时间,单位秒! 默认: -1 永不超时,不限制事务时间!*/Transactional(readOnly false,timeout 3)public void changeInfo(){studentDao.updateAgeById(100,1);//休眠4秒,等待方法超时!try {Thread.sleep(4000);} catch (InterruptedException e) {throw new RuntimeException(e);}studentDao.updateNameById(test1,1);}
}
5、只读
对一个查询操作来说如果我们把它设置成只读就能够明确告诉数据库这个操作不涉及写操作。这样数据库就能够针对查询操作来进行优化。
设置方式
// readOnly true把当前事务设置为只读 默认是false!
Transactional(readOnly true)针对DML动作设置只读模式
会抛出下面异常
Caused by: java.sql.SQLException: Connection is read-only. Queries leading to data modification are not allowed