Spring 事务原理和使用

转载自  Spring 事务原理和使用

1.Spring@Transactional的配置

步骤一、在Spring配置文件中引入命名空间

<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:tx="http://www.springframework.org/schema/tx"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsdhttp://www.springframework.org/schema/txhttp://www.springframework.org/schema/tx/spring-tx-2.0.xsd">

步骤二、xml配置文件中,添加事务管理器bean配置

<!-- 事务管理器配置,单数据源事务 --><bean id="pkgouTransactionManager"class="org.springframework.jdbc.datasource.DataSourceTransactionManager"><property name="dataSource" ref="pkGouDataSource" /></bean>
<!-- 使用annotation定义事务 --><tx:annotation-driven transaction-manager="pkgouTransactionManager" />

步骤三、在使用事务的方法或者类上添加下面的注解

@Transactional(“pkgouTransactionManager”)

 

2.传播行为和隔离级别

1> 事务注解方式: @Transactional

  • 标注在类前:标示类中所有方法都进行事务处理

  • 标注在接口、实现类的方法前:标示方法进行事务处理

2> 事务传播行为介绍:

 

3> 事务超时设置:

@Transactional(timeout=30) //默认是30秒

4> 事务隔离级别:

 

  • 脏读 : 一个事务读取到另一事务未提交的更新数据

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

  • 幻读 : 一个事务读到另一个事务已提交的insert数据

@Transactional的属性:

 

3.工作原理

自动提交

默认情况下,数据库处于自动提交模式。每一条语句处于一个单独的事务中,在这条语句执行完毕时,如果执行成功则隐式的提交事务,如果执行失败则隐式的回滚事务。 事务管理,是一组相关的操作处于一个事务之中,因此必须关闭数据库的自动提交模式。这点,Spring会在org/springframework/jdbc/datasource/DataSourceTransactionManager.java中将底层连接的自动提交特性设置为false。

// switch to manual commit if necessary。 this is very expensive in some jdbc drivers,
// so we don't want to do it unnecessarily (for example if we've explicitly
// configured the connection pool to set it already)。if (con。getautocommit()) 
{txobject.setmustrestoreautocommit(true);if (logger.isdebugenabled()) {logger.debug("switching jdbc connection [" + con + "] to manual commit");}//首先将自动提交属性改为falsecon.setautocommit(false);
}

spring事务回滚规则

Spring事务管理器回滚一个事务的推荐方法是在当前事务的上下文内抛出异常。Spring事务管理器会捕捉任何未处理的异常,然后依据规则决定是否回滚抛出异常的事务。 默认配置下,Spring只有在抛出的异常为运行时unchecked异常时才回滚该事务,也就是抛出的异常为RuntimeException的子类(Errors也会导致事务回滚)。而抛出checked异常则不会导致事务回滚。 Spring也支持明确的配置在抛出哪些异常时回滚事务,包括checked异常。也可以明确定义哪些异常抛出时不回滚事务。 还可以编程性的通过setRollbackOnly()方法来指示一个事务必须回滚,在调用完setRollbackOnly()后你所能执行的唯一操作就是回滚。

4.注意事项

由于Spring事务管理是基于接口代理或动态字节码技术,通过AOP实施事务增强的。

(1)对于基于接口动态代理的AOP事务增强来说,由于接口的方法是public的,这就要求实现类的实现方法必须是public的(不能是protected,private等),同时不能使用static的修饰符。所以,可以实施接口动态代理的方法只能是使用“public” 或 “public final”修饰符的方法,其它方法不可能被动态代理,相应的也就不能实施AOP增强,也即不能进行Spring事务增强。

(2)基于CGLib字节码动态代理的方案是通过扩展被增强类,动态创建子类的方式进行AOP增强植入的。由于使用final,static,private修饰符的方法都不能被子类覆盖,相应的,这些方法将不能被实施的AOP增强。

所以,必须特别注意这些修饰符的使用,@Transactional 注解只被应用到 public 可见度的方法上。 如果你在 protected、private 或者 package-visible 的方法上使用 @Transactional 注解,它也不会报错,但是这个被注解的方法将不会展示已配置的事务设置。

用 spring 事务管理器,由spring来负责数据库的打开,提交,回滚。默认遇到运行期异常(throw new RuntimeException(“注释”);)会回滚,即遇到不受检查(unchecked)的异常时回滚;

@Transactional(rollbackFor=Exception.class) //指定回滚,遇到异常Exception时回滚
public void methodName() 
{
throw new Exception("注释");
}

而遇到需要捕获的异常(throw new Exception(“注释”);)不会回滚,即遇到受检查的异常(就是非运行时抛出的异常,编译器会检查到的异常叫受检查异常或说受检查异常)时,需我们指定方式来让事务回滚 要想所有异常都回滚,要加上 @Transactional(rollbackFor={Exception。class,其它异常}) 。如果让unchecked异常不回滚:

@Transactional(notRollbackFor=RunTimeException.class)如下:@Transactional(noRollbackFor=Exception.class)//指定不回滚,遇到运行期异常(throw new RuntimeException("注释");)会回滚
public ItimDaoImpl getItemDaoImpl() 
{
throw new RuntimeException("注释");
}

仅仅 @Transactional注解的出现不足于开启事务行为,它仅仅是一种元数据,能够被可以识别 @Transactional注解和上述的配置适当的具有事务行为的beans所使用。其实,根本上是 元素的出现 开启了事务行为。

Spring团队的建议是你在具体的类(或类的方法)上使用 @Transactional 注解,而不要使用在类所要实现的任何接口上。你当然可以在接口上使用 @Transactional 注解,但是这将只能当你设置了基于接口的代理时它才生效。因为注解是不能继承的,这就意味着如果你正在使用基于类的代理时,那么事务的设置将不能被基于类的代理所识别,而且对象也将不会被事务代理所包装(将被确认为严重的)。因此,请接受Spring团队的建议并且在具体的类火方法上使用 @Transactional 注解。

@Transactional 注解标识的方法,处理过程尽量的简单。尤其是带锁的事务方法,能不放在事务里面的最好不要放在事务里面。可以将常规的数据库查询操作放在事务前面进行,而事务内进行增、删、改、加锁查询等操作。

@Transactional 注解的默认事务管理器bean是“transactionManager”,如果声明为其他名称的事务管理器,需要在方法上添加@Transational(“managerName”)。

@Transactional 注解标注的方法中不要出现网络调用、比较耗时的处理程序,因为,事务中数据库连接是不会释放的,如果每个事务的处理时间都非常长,那么宝贵的数据库连接资源将很快被耗尽。

5.自我调用中的问题 

Spring事务使用AOP代理后的方法调用执行流程,如图所示:

 

从图中可以看出,调用事务时首先调用的是AOP代理对象而不是目标对象,首先执行事务切面,事务切面内部通过TransactionInterceptor环绕增强进行事务的增强。即进入目标方法之前开启事务,退出目标方法时提交/回滚事务。

这样在自我调用时,则会出现无法开启事务的问题,比如:

public interface TargetService 
{  public void a();  public void b();  
}  
@Service 
public class TargetServiceImpl implements TargetService
{  public void a() {  this.b();  }  @Transactional(propagation = Propagation.REQUIRES_NEW)  public void b() {//执行数据库操作}  
}

此处的this指向目标对象,因此调用this.b()将不会执行b事务切面,即不会执行事务增强,因此b方法的事务定义“@Transactional(propagation = Propagation.REQUIRES_NEW)”将不会实施,即结果是b和a方法的事务是方法的事务定义是一样的。

解决方法 通过BeanPostProcessor 在目标对象中注入代理对象:

一、定义BeanPostProcessor 需要使用的标识接口

public interface BeanSelfAware
{public abstract void setSelf(Object obj);
}

二、定义自己的BeanPostProcessor(InjectBeanSelfProcessor)

public class InjectBeanSelfProcessorimplements BeanPostProcessor, ApplicationContextAware
{ApplicationContext context;private static Log log = LogFactory.getLog(com/netease/lottery/base/common/BeanSelf/InjectBeanSelfProcessor);
public InjectBeanSelfProcessor()
{
}
public void setApplicationContext(ApplicationContext context)throws BeansException
{this.context = context;
}
public Object postProcessAfterInitialization(Object bean, String beanName)throws BeansException
{if(bean instanceof BeanSelfAware){//如果Bean实现了BeanSelfAware标识接口,就将代理对象注入BeanSelfAware myBean = (BeanSelfAware)bean;Class cls = bean.getClass();if(!AopUtils.isAopProxy(bean)){Class c = bean.getClass();Service serviceAnnotation = (Service)c.getAnnotation(org/springframework/stereotype/Service);if(serviceAnnotation != null)try{bean = context.getBean(beanName);if(AopUtils.isAopProxy(bean));}catch(BeanCurrentlyInCreationException beancurrentlyincreationexception) { }catch(Exception ex){log.fatal((new StringBuilder()).append("No Proxy Bean for service ").append(bean.getClass()).append(" ").append(ex.getMessage()).toString(), ex);}}myBean.setSelf(bean);return myBean;} else{return bean;}
}
public Object postProcessBeforeInitialization(Object bean, String beanName)throws BeansException
{return bean;
}

三、目标类实现

public interface TargetService 
{  
public void a();  
public void b();  
}  
@Service 
public class TargetServiceImpl implements TargetService,BeanSelfAware
{  
private TargetService self;  
public void setSelf(Object proxyBean) 
{ //通过InjectBeanSelfProcessor注入自己(目标对象)的AOP代理对象  this.self = (TargetService) proxyBean;  
}  
public void a() 
{  self.b();  
}  
@Transactional(propagation = Propagation.REQUIRES_NEW)  
public void b() 
{
//执行数据库操作
}  
}

postProcessAfterInitialization根据目标对象是否实现BeanSelfAware标识接口,通过setSelf(bean)将代理对象(bean)注入到目标对象中,从而可以完成目标对象内部的自我调用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/326110.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

微软在.NET官网上线.NET 架构指南频道

微软在Visual Studio 2017 正式发布的时候也上线了一个参考应用https://github.com/dotnet/eShopOnContainers , 最近微软给这个参考应用写了完善的文档&#xff0c;放在.NET官网的.NET架构频道https://www.microsoft.com/net/architecture。 整个.NET 架构按照4个部分展开&am…

2015蓝桥杯省赛---java---B---10(生命之树)

题目描述 思路分析 dfs无根树变有根树 代码实现 package lanqiao;import java.util.Scanner;public class Main {//提升作用域public static int n;public static int[] w;//每个点的权重public static int[] ww;//每个点作为根节点时能得到的最大权和public static int[][…

wps工资表怎么用计算机,wps制作工资表的方法步骤图

wps是一款强大的日常办公软件&#xff0c;我们可以使用wps来制作工资表&#xff0c;想知道具体的操作吗?下面就让学习啦小编告诉你wps如何制作工资表&#xff0c;希望对大家有所帮助。wps制作工资表的方法第一节 建立与使用 WPS表格WPS Office 2007办公 软件中提供了一个名为 …

Java8面试题

转载自 Java8面试题 问题一&#xff1a; Java8支持函数编程是什么意思&#xff1f; 在Java 8之前&#xff0c;所有东西都是面向对象的。除了原语之外&#xff0c;java中的 所有内容都作为对象存在。对方法/函数的所有调用都是使用对象或类引用进行的。 方法/功能本身并不…

为什么转换到Visual Studio 2017如此 “容易”

与VS 2015相比&#xff0c;Visual Studio 2017在C 功能方面有重大的飞跃。我们希望升级到新版本后能让您的日常工作变得更轻松。 这篇文章主要介绍从Visual Studio 2015升级到2017的步骤。在这个版本中&#xff0c;我们团队使代码库移动到Visual Studio 2017非常容易。这里有四…

2015蓝桥杯省赛---java---A---9(垒筛子)

题目描述 思路分析 递归 会超时 通过30% 代码实现 package lanqiao;import java.util.Scanner;public class Main {public static int op[]new int[7];public static int n,m;public static boolean conflict[][]new boolean[7][7];public static int MOD1000000007;publi…

想进入谷歌吗?先来看看这些面试题吧

转载自 想进入谷歌吗&#xff1f;先来看看这些面试题吧 谷歌&#xff0c;美国的跨国科技企业&#xff0c;致力于互联网搜索、云计算、广告技术等领域&#xff0c;开发并提供大量基于互联网的产品与服务。2016年6月8日&#xff0c;《2016年BrandZ全球最具价值品牌百强榜》公布…

计算机沙盒应用,既能防木马病毒又能多开应用,这款软件Windows不能没有

各位差友平时在 Windows 电脑上是怎么下载软件的呢&#xff1f;世超猜测很多人会选择在浏览器里直接搜索某个软件的名字&#xff0c;搜索结果中出现官网就从官网下载&#xff0c;没有官网就进到各种资源聚合网站碰碰运气。而众所周知&#xff0c;Windows 平台向来不缺各种流氓软…

从真实项目中抠出来的设计模式——第三篇:责任链模式

一&#xff1a;现实场景 有时候在开发的过程中&#xff0c;我们经常会根据某个状态的值&#xff0c;写出很多的ifelse逻辑&#xff0c;比如拿项目里面的案例来说&#xff0c;如果当前发送的是彩信&#xff0c;此种状态需要如何给实体赋值&#xff0c;如果是短信&#xff0c;邮件…

Spark之Master主备切换机制原理

Spark之Master主备切换机制原理

SQL正在击败NoSQL,这对未来的数据意味着什么

转载自 SQL正在击败NoSQL&#xff0c;这对未来的数据意味着什么 经过多年的沉寂之后&#xff0c;今天的SQL正在复出。缘由如何&#xff1f; 这对数据社区有什么影响&#xff1f; 自从可以利用计算机做事以来&#xff0c;我们一直在收集的数据以指数级的速度在增长&#xff0…

公共计算机课程思政建设实施方案,公共计算机教研组开设课程思政公开课

为了进一步奠定课程思政基础,提高教师将思政元素融入到专业课程的能力,创造一个课程思政的良好氛围,2021年4月30日上午&#xff0c;公共计算机教研组陈雪云老师在文虎楼401教室开设《计算机文化基础(理)》思政课程公开课。授课班级是2020级数学师范1、2班。李美莲、黄婷、龚长斌…

数字奇数和偶数的判断

package lanqiao;/*** 创建人 wdl* 创建时间 2021/4/16* 描述*/ public class TEST {public static void main(String[] args) {int a7,b8;if((a&1)1){System.out.println("a是奇数");}if((b&1)0){System.out.println("b是偶数");}}}

asp.net core源码飘香:从Hosting开始

知识点&#xff1a; 1.Kestrel服务器启动并处理Http请求的过程。 2.Startup的作用。 源码飘香&#xff1a; 总结&#xff1a; asp.net core将web开发拆分为多个独立的组件&#xff0c;大多以http中间件的形式添加到请求管线&#xff0c;也有一下作为基础组件&#xff08;con…

这些Spring中的设计模式,你都知道吗?

转载自 这些Spring中的设计模式&#xff0c;你都知道吗&#xff1f; 设计模式作为工作学习中的枕边书&#xff0c;却时常处于勤说不用的尴尬境地&#xff0c;也不是我们时常忘记&#xff0c;只是一直没有记忆。 Spring作为业界的经典框架&#xff0c;无论是在架构设计方面&a…

如何移植.NET Framework项目至.NET Core?

公司的项目一直采用.NET框架来开发Web项目。目前基础类库均为.NET Framework 4.6.2版本。Caching, Logging&#xff0c;DependencyInjection&#xff0c;Configuration等基础设施相关的依赖库一直和官方保持同步&#xff0c;目前是1.1版本。.NET Core越来越趋于稳定&#xff0c…

2016蓝桥杯省赛---java---A---8(取球博弈)

题目描述 思路分析 dfs 代码实现 package lanqiao;import java.util.Arrays; import java.util.Scanner;public class Main {public static int []n;public static void main(String[] args) {Scanner scanner new Scanner(System.in);nnew int[3];for (int i 0; i < 3…

分布式统一配置平台-Disconf.Net

为了更好的解决分布式环境下多台服务实例的配置统一管理问题&#xff0c;本文提出了一套完整的分布式配置管理解决方案。结合.net项目具体情况&#xff0c;实现了配置发布的统一化&#xff0c;对配置进行持久化管理并对外提供restful接口&#xff0c;在此基础上&#xff0c;基于…

计算机软件录音注意事项,GOLDWAVE录音软件使用教程

很多朋友都有录音软件&#xff0c;但是软件功能不全&#xff0c;也不是专业的&#xff0c;所以现在我讲一个专业的软件&#xff0c;这个软件都是原创歌手比较常用的软件&#xff0c;现在我们来学习一下。这是该软件的界面&#xff0c;软件的名字叫做GOLDWAVE这款软件的功能很强…

2016蓝桥杯省赛---java---A---10(压缩转换)

题目描述 小明最近在研究压缩算法。 他知道&#xff0c;压缩的时候如果能够使得数值很小&#xff0c;就能通过熵编码得到较高的压缩比。 然而&#xff0c;要使数值很小是一个挑战。 最近&#xff0c;小明需要压缩一些正整数的序列&#xff0c;这些序列的特点是&#xff0c;后面…