Spring Data JPA 从入门到精通~事务的处理及其讲解

默认 @Transactional 注解式事务

(1)@EnableTransactionManagement

正常情况下,我们是需要在 ApplicationConfig 类加上 @EnableTransactionManagement 注解才能开启事务管理。通过 DataSource 的研究步骤 spring.factories 里面默认加载 TransactionAutoConfiguration 类,而我们看源码,其里面已经加了此注解,默认采用的 AdviceMode.PROXY,所以默认情况的事务管理机制是代理方式的,通过添加 @Transactional 注解式配置方法,查看 SimpleJpaRepository:

@Repository
@Transactional(readOnly = true)
public class SimpleJpaRepository<T, ID> implements JpaRepository<T, ID>, JpaSpecificationExecutor<T> {...}

所以每个 Respository 的方法是都是有默认的只读事务的。

(2)我们来查看一下 @Transactional 源码:

@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @interface Transactional {@AliasFor("transactionManager")String value() default "";@AliasFor("value")String transactionManager() default "";Propagation propagation() default Propagation.REQUIRED;Isolation isolation() default Isolation.DEFAULT;int timeout() default TransactionDefinition.TIMEOUT_DEFAULT;boolean readOnly() default false;Class<? extends Throwable>[] rollbackFor() default {};String[] rollbackForClassName() default {};Class<? extends Throwable>[] noRollbackFor() default {};String[] noRollbackForClassName() default {};
}

@Transactional 注解中常用参数说明

参数名称

功能描述

readOnly

rollbackFor

rollbackForClassName

noRollbackFor

noRollbackForClassName

propagation

isolation

timeout

transactionManager/value

(3)propagation:传播行为

所谓事务的传播行为是指,如果在开始当前事务之前,一个事务上下文已经存在,此时有若干选项可以指定一个事务性方法的执行行为。

可以看 org.springframework.transaction.annotation.Propagation 枚举类中定义了 7 个表示传播行为的枚举值:

public enum Propagation { REQUIRED(0), SUPPORTS(1), MANDATORY(2), REQUIRES_NEW(3), NOT_SUPPORTED(4), NEVER(5), NESTED(6); }

  • REQUIRED:如果当前存在事务,则加入该事务;如果当前没有事务,则创建一个新的事务。
  • SUPPORTS:如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
  • MANDATORY:如果当前存在事务,则加入该事务;如果当前没有事务,则抛出异常。
  • REQUIRES_NEW:创建一个新的事务,如果当前存在事务,则把当前事务挂起。
  • NOT_SUPPORTED:以非事务方式运行,如果当前存在事务,则把当前事务挂起。
  • NEVER:以非事务方式运行,如果当前存在事务,则抛出异常。
  • NESTED:如果当前存在事务,则创建一个事务作为当前事务的嵌套事务来运行;如果当前没有事务,则该取值等价于REQUIRED。

设置方法:通过使用propagation属性设置,例如:

@Transactional(propagation = Propagation.REQUIRED)

(4)Isolation 隔离级别

隔离级别是指若干个并发的事务之间的隔离程度,与我们开发时候主要相关的场景包括:脏读取、重复读、幻读。

我们可以看 org.springframework.transaction.annotation.Isolation 枚举类中定义了四个表示隔离级别的值:

public enum Propagation {REQUIRED(0),SUPPORTS(1),MANDATORY(2),REQUIRES_NEW(3),NOT_SUPPORTED(4),NEVER(5),NESTED(6);
}
  • DEFAULT:这是默认值,表示使用底层数据库的默认隔离级别,对大部分数据库而言,通常这值就是READ_COMMITTED。
  • READ_UNCOMMITTED:该隔离级别表示一个事务可以读取另一个事务修改但还没有提交的数据。该级别不能防止脏读和不可重复读,因此很少使用该隔离级别。
  • READ_COMMITTED:该隔离级别表示一个事务只能读取另一个事务已经提交的数据。该级别可以防止脏读,这也是大多数情况下的推荐值。
  • REPEATABLE_READ:该隔离级别表示一个事务在整个过程中可以多次重复执行某个查询,并且每次返回的记录都相同。即使在多次查询之间有新增的数据满足该查询,这些新增的记录也会被忽略。该级别可以防止脏读和不可重复读。
  • SERIALIZABLE:所有的事务依次逐个执行,这样事务之间就完全不可能产生干扰,也就是说,该级别可以防止脏读、不可重复读以及幻读,但是这将严重影响程序的性能。通常情况下也不会用到该级别。

设定方法为通过使用 isolation 属性设置,例如:

@Transactional(isolation = Isolation.DEFAULT)

(5)所以 Spring Boot 的这种默认机制,只需要在我们用事务时,在方法上或者此方法的类上加上 @Transactional注解即可。

而实际工作中,我们一般都要在 Service 层的某些方法上加事务,以保证整个方法的事务。示例如下:

@Transactional(rollbackOn = Exception.class)
public void saveUserInfo() throws Exception {userCustomerRepository.save(new UserCustomerEntity("jackzhang@mail.com","jackzhang"));userRepository.save(new UserInfoEntity("jack_test","name"));throw new Exception("test");......//此方法体有多个repository的调用,模拟异常,事务会回滚的
}

(6)注意的几点

  • @Transactional 只能被应用到 public 方法上,对于其他非 public 的方法,如果标记了 @Transactional 也不会报错,但方法没有事务功能。
  • 用 spring 事务管理器,由 spring 来负责数据库的打开、提交、回滚,默认遇到运行期例外(throw new RuntimeException("注释");)会回滚,即遇到不受检查(unchecked)的例外时回滚;而遇到需要捕获的例外(throw new Exception("注释");)不会回滚,即遇到受检查的例外(就是非运行时抛出的异常,编译器会检查到的异常叫受检查例外或说受检查异常)时,需我们指定方式来让事务回滚要想所有异常都回滚,要加上 @Transactional( rollbackFor={Exception.class,其他异常}),如果让 unchecked 例外不回滚:@Transactional(notRollbackFor=RunTimeException.class)。
  • @Transactional 注解应该只被应用到 public 可见度的方法上。如果你在 protected、private 或者 package-visible 的方法上使用 @Transactional 注解,它也不会报错,但是这个被注解的方法将不会展示已配置的事务设置。
  • @Transactional 注解可以被应用于接口定义和接口方法、类定义和类的 public 方法上。然而,请注意仅仅 @Transactional 注解的出现不足于开启事务行为,它仅仅是一种元数据,能够被可以识别 @Transactional 注解和上述的配置适当的具有事务行为的 beans 所使用。上面的例子中,其实正是 元素的出现开启了事务行为。
  • Spring 团队的建议是你在具体的类(或类的方法)上使用 @Transactional 注解,而不要使用在类所要实现的任何接口上。当然可以在接口上使用 @Transactional 注解,但是这将只能当你设置了基于接口的代理时它才生效。因为注解是不能继承的,这就意味着如果你正在使用基于类的代理时,那么事务的设置将不能被基于类的代理所识别,而且对象也将不会被事务代理所包装(将被确认为严重的)。因此,请接受 Spring 团队的建议并且在具体的类上使用 @Transactional 注解。
  • 事务有两种配置方法,一种是我们现在说的显式的注解式事务,当我们注解式事务下,不加注解 service 方法上是没有任何事务的。还有一种是隐式事务,ASPECTJ 的思路配置方法,所以不是没有加 @Transactional 注解就一定没有事务。

声明式事务,又叫隐式事务,或者叫 ASPECTJ 事务

配置方法:

在实际工作中,每个方法都让我们加上 @Transactional 注解,可能工作量有点大,也有时候会忘,所以经常看到有开发团队配置拦截式事务。虽然 spring 官方不太推荐。只需要在我们的项目中新增一个类 AspectjTransactionConfig 即可,如下:

@Configuration
@EnableTransactionManagement
public class AspectjTransactionConfig {public static final String transactionExecution = "execution (* com.jackzhang.example..service.*.*(..))";@Autowiredprivate PlatformTransactionManager transactionManager;@Beanpublic DefaultPointcutAdvisor defaultPointcutAdvisor() {//指定一般要拦截哪些类AspectJExpressionPointcut pointcut = new AspectJExpressionPointcut();pointcut.setExpression(transactionExecution);//配置advisorDefaultPointcutAdvisor advisor = new DefaultPointcutAdvisor();advisor.setPointcut(pointcut);//指定不同的方法用不通的策略Properties attributes = new Properties();attributes.setProperty("get*", "PROPAGATION_REQUIRED,-Exception");attributes.setProperty("add*", "PROPAGATION_REQUIRED,-Exception");attributes.setProperty("save*", "PROPAGATION_REQUIRED,-Exception");attributes.setProperty("update*", "PROPAGATION_REQUIRED,-Exception");attributes.setProperty("delete*", "PROPAGATION_REQUIRED,-Exception");//创建InterceptorTransactionInterceptor txAdvice = new TransactionInterceptor(transactionManager, attributes);advisor.setAdvice(txAdvice);return advisor;}
}

这样我们的 Service 就会自动拥有了事务,可以加 @Transactional 来覆盖全局的配置。

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

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

相关文章

ASP.NET MVC V2 Preview 1 发布 期望VS有更好的表现

ASP.NET MVC V2 Preview 1官方首页&#xff1a;http://aspnet.codeplex.com/ 在这里可以下载 以下是网友的转载&#xff0c;介绍的还是比较详细的&#xff1a; 预览版是在.NET 3.5 SP1和VS 2008下工作的&#xff0c;可与ASP.NET MVC 1.0并行安装在同一个机器上&#xff08;即&a…

MC缓存序列化php,PHP serialize()序列化的使用

PHP serialize()序列化的使用可以将数组和对象直接存入数据库中的某一字段。使serialize()是将数组反序列化再存入数据库&#xff0c;序列化话完的数据就是一个字符串。提取的时候&#xff0c;用unserialize()反序列化取&#xff0c;取出来的还是个数组。$arr array(value1,va…

全球最权威的区块链行业报告

来源&#xff1a;腾讯研究院美国加密货币报道媒体CoinDesk近期发布“全球区块链现状报告”&#xff0c;深入研究了快速发展的加密货币行业及其底层技术&#xff0c;该报告覆盖了公共区块链、企业区块链、ICO、投资以及监管等话题&#xff0c;另外还对3000多名投资者的加密货币投…

Google 的 C++ 代码规范

Google 开源项目风格指南 (中文版)&#xff1a;https://zh-google-styleguide.readthedocs.io/en/latest/ 英文版&#xff1a;http://google-styleguide.googlecode.com/svn/trunk/cppguide.xml 中文版&#xff1a;http://zh-google-styleguide.readthedocs.org/en/latest/goog…

解析/etc/inittab 文件(转)

原文地址&#xff1a;http://hi.baidu.com/fembed/blog/item/62a509f01b2e69aaa40f528c.htmlinit程序需要读取配置文件/etc/inittab。inittab是一个不可执行的文本文件&#xff0c;它有若干行指令所组成。在Redhat系统中&#xff0c;inittab的内容如下所示(以“###"开始的…

Spring Data JPA 从入门到精通~如何配置多数据源

如何配置多数据源 在 application.properties 中定义两个 DataSource 定义两个 DataSource 用来读取 application.properties 中的不同配置。如下例子中&#xff0c;主数据源配置为 spring.datasource.one 开头的配置&#xff0c;第二数据源配置为 spring.datasource.two 开头…

NASA投资有远景技术,有望改变未来人类和机器人的勘探任务

来源&#xff1a; 机器人创新生态据NASA官网报道&#xff0c;美国宇航局(NASA)正在投资有远见的技术概念&#xff0c;包括流星撞击探测、太空望远镜群以及细小轨道碎片测绘技术&#xff0c;这些技术将来可能被用于未来的太空探索任务中。美国宇航局已经选出25个还处于早期的技术…

feedback.php,feedback.php

Feedback | Training Labif (!isset($_SESSION[username])) {header(Location: index.php#loginSub);}if($_GET){//查询require_once DAO.php;$dbnew DB();$data[fb_id] $_GET[feedbackid];$judge[fb_id] ;list($conSql, $mapConData) $db->FDFields($data,"",…

2008中国软件企业出口(外包)20强

2008中国软件企业出口20强 1 东软集团股份有限公司 2 大连华信计算机技术股份有限公司 3 海辉软件(国际)集团 4 浙大网新科技股份有限公司 5 中软国际有限公司 6 北大方正集团有限公司 7 上海中和软件有限公司 8 日电卓越软件科技(北京)有限公司 9 上海海隆软件股份有限公司 10…

人工智能如何影响社会经济:关于近期文献的一个综述

作者&#xff1a;陈永伟 文章来源&#xff1a;经济学原理 相比于之前的历次技术进步&#xff0c;“人工智能革命”所引发的冲击更为巨大&#xff0c;其对经济学造成的影响也将更为广泛和深远。人工智能技术的突飞猛进&#xff0c;对经济社会的各个领域都产生了重大影响&#…

oracle近似查找,距离内的Oracle空间搜索

对于mySQL距离搜索,你有很好的参考.忘记Oracle Spatial的东西.代码太多,复杂性太大,没有足够的增值.这是一个可以解决问题的查询.这使用法定里程的距离.编辑这修复了mdarwin提到的错误,如果你试图将它用于北极或南极的位置,则以划分检查为代价.SELECT id, city, LATITUDE, LONG…

Spring Data JPA 从入门到精通~Naming命名策略详解及其实践

Naming 命名策略详解及其实践 用 JPA 离不开 Entity 实体&#xff0c;我都知道实体里面有字段映射&#xff0c;而字段映射的方法有两种&#xff1a; 显式命名&#xff1a;在映射配置时&#xff0c;设置的数据库表名、列名等&#xff0c;就是进行显式命名&#xff0c;即通过 C…

C 和 C++ 字符串格式化

--------------------------------------------------------------------------------- gcc中格式化输出函数的研究&#xff1a;http://blog.csdn.net/dai_xiangjun/article/details/41356723 C语言格式化输出小结&#xff1a;http://blog.51cto.com/happyliu/1621709 代码如…

转载(四).Net Framework中的委托与事件

尽管 &#xff08;委托、事件与Observer设计模式 &#xff09;中的范例很好地完成了我们想要完成的工作&#xff0c;但是我们不仅疑惑&#xff1a;为什么.Net Framework 中的事件模型和上面的不同&#xff1f;为什么有很多的EventArgs参数&#xff1f; 在回答上面的问题之前&am…

激光雷达:从光电技术角度看自动驾驶

来源&#xff1a; 传感器技术激光雷达和与之竞争的传感器技术&#xff08;相机、雷达和超声波&#xff09;加强了对传感器融合的需要&#xff0c;也对认真谨慎地选择光电探测器、光源和MEMS振镜提出了更高的要求。传感器技术、成像、雷达、光探测技术及测距技术&#xff08;激光…

Spring Data JPA 从入门到精通~定义查询方法的配置方法

定义查询方法的配置方法 由于 Spring JPA Repository 的实现原理是采用动态代理的机制&#xff0c;所以我们介绍两种定义查询方法&#xff0c;从方法名称中可以指定特定用于存储的查询和更新&#xff0c;或通过使用 Query 手动定义的查询&#xff0c;取决于实际对数据的操作&a…

socket的长连接、短连接、半包、粘包与分包

socket的半包&#xff0c;粘包与分包的问题和处理代码&#xff1a;http://blog.csdn.net/qq_16112417/article/details/50392463 知乎关于长连接和短连接&#xff1a;https://www.zhihu.com/search?typecontent&q长连接%20短连接 TCP网络通讯如何解决分包粘包问题&#…

在Windows2003server上安装msn9.0以上版本的解决办法

方法一&#xff1a; 最近windows 强制升级了msn 9的版本。导致我在windows2003下的msn又一次需要剥离出msi文件才能安装了。在网上逛了一圈发现了如下方法&#xff0c;这里分享下&#xff0c;同时&#xff0c;也感谢作者的详细步骤。 2003SERVER版安装MSN 9.0&#xff08;版本2…

oracle exec编译失效,编译oracle失效的函数、存储过程、视图等

进行oracle开发时常常为不时失效的存储过程&#xff0c;函数等苦恼&#xff0c;于是下决心解决这个问题&#xff0c;东查西找&#xff0c;找到一个好东东&#xff0c;在基础上又修改了一下,问题终于被我解决了。1。建立一个存储过程CREATE OR REPLACE PROCEDURE TIMER_AUTO_REC…

2018年中国人工智能行业研究报告|附下载

来源&#xff1a;网络大数据、艾瑞咨询广义人工智能指通过计算机实现人的头脑思维所产生的效果&#xff0c;是对能够从环境中获取感知并执行行动的智能体的描述和构建;相对狭义的人工智能包括人工智能产业(包含技术、算法、应用等多方面的价值体系)、人工智能技术(包括凡是使用…