事务传播行为_第1页
事务传播行为_第2页
事务传播行为_第3页
事务传播行为_第4页
事务传播行为_第5页
已阅读5页,还剩4页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、事务传播行为文章一:我们都知道事务的概念,那么事务的传播特性是什么呢? (此处着重介绍传播特性的概念, 关于传播特性的 相关配置就不介绍了,可以查看 spring 的官方文档 ) 在我们 用 SSH 开发项目的时候, 我们一般都是将事务设置在 Service层 那么当我们调用 Service 层的一个方法的时候它能够保证 我们的这个方法中执行的所有的对数据库的更新操作保持 在一个事务中,在事务层里面调用的这些方法要么全部成Dao 层功,要么全部失败。 那么事务的传播特性也是从这里说起的。如果你在你的 Service 层的这个方法中,除了调用了 的方法之外,还调用了本类的其他的 Service

2、方法,那么在 调用其他的 Service 方法的时候,这个事务是怎么规定的呢, 我必须保证我在我方法里掉用的这个方法与我本身的方法 处在同一个事务中,否则如果保证事物的一致性。事务的传 播特性就是解决这个问题的, “事务是会传播的”在 Spring中有针对传播特性的多种配置我们大多数情况下只用其 的一种 :PROPGATION_REQUIRED :这个配置项的意思是说 当我调用 service 层的方法的时候开启一个事务 (具体调用那层的方法开始创建事务,要看你的aop的配置),那么在调用这个 service 层里面的其他的方法的时候 ,如果当前方法产生了事务就用当前方法产生的事务,否则就创建

3、一个新的事务。这个工作使由 Spring 来帮助我们完成的。 以前没有Spring 帮助我们完成事务的时候我们必须自己手动的控制事务,例如当我们项目中仅仅使用hibernate,而没有集成进spring 的时候,我们在一个 service 层中调用其他的业务逻辑方法,为了保证事物必须也要把当前的hibernate session 传递session到下一个方法中,或者采用 ThreadLocal 的方法,将 传递给下一个方法,其实都是一个目的。现在这个工作由spring 来帮助我们完成,就可以让我们更加的专注于我们的 业务逻辑。而不用去关心事务的问题。 默认情况下当发生RuntimeExcep

4、tion 的情况下,事务才会回滚,所以要注意如果你在程序发生错误的情况下,有自己的异常处理机 制定义自己的 Exception ,必须从 RuntimeException 类继承 这 样事务才会回滚!Propagation : key文章二: 事务的传播行为和隔离级别 transaction behavior and中事务的定义:isolated levelSpring属性确定代理应该给哪个方法增加事务行为。这样的属性最 重要的部份是传播行为。有以下选项可供使用:PROPAGATION_REQUIRED- 支持当前事务,如果当前没有 事务,就新建一个事务。这是最常见的选择。PROPAGATIO

5、N_SUPPORTS- 支持当前事务,如果当前没有 事务,就以非事务方式执行。PROPAGATION_MANDATORY- 支持当前事务, 如果当前没有事务,就抛出异常。 PROPAGATION_REQUIRES_NEW-新建事务,如果当前存在事务,把当前事务挂起。PROPAGATION_NOT_SUPPORTED- 以非事务方式执行操 作,如果当前存在事务,就把当前事务挂起。PROPAGATION_NEVER- 以非事务方式执行, 如果当前存在 事务,则抛出异常。 很多人看到事务的传播行为属性都不 甚了解,我昨晚看了 j2ee without ejb 的时候,看到这里也不 了解,甚至重新翻起

6、数据库系统的教材书,但是也没有找到 对这个的分析。今天搜索,找到一篇极好的分析文章,虽然 这篇文章是重点分析 PROPAGATION_REQUIRED 和PROPAGATION_REQUIRED_NESTED 的解惑 spring 嵌套事务 *TransactionDefinition 接口定义*int PROPAGATION_REQUIRED= 0;int PROPAGATION_SUPPORTS = 1;int PROPAGATION_MANDATORY = 2;intPROPAGATION_REQUIRES_NEW = 3;intPROPAGATION_NOT_SUPPORTED = 4

7、;int PROPAGATION_NEVER = 5;intPROPAGATION_NESTED = 6;*在这篇文章里,他用两个嵌套的例子辅助分析,我这里直接引用了。*sample*ServiceA void methodA()ServiceB.methodB();ServiceB void methodB()*们这里一个个分析吧 1: PROPAGATION_REQUIRED加入当前正要执行的事务不在另外一个事务里,那么就起一个新的事务比如说, ServiceB.methodB 的事务级别定义为PROPAGATION_REQUIRED, 那么由于执行ServiceA.methodA 的时候

8、,ServiceA.methodA已经起了事务,这时调用 ServiceB.methodB , ServiceB.methodB看到自己已经运行在 ServiceA.methodA 的事务内部, 就不再 起新的事务。 而假如 ServiceA.methodA 运行的时候发现自己 没有在事务中,他就会为自己分配一个事务。这样,在ServiceA.methodA 或者在 ServiceB.methodB 内的任何地方出 现异常,事务都会被回滚。即使 ServiceB.methodB 的事务已 经被提交,但是 ServiceA.methodA 在接下来 fail 要回滚,ServiceB.metho

9、dB 也要回滚 2:PROPAGATION_SUPPORTS 如果当前在事务中,即以事务的形式运行,如果当前不再一个事务中,那么就以非事务的形式运行这就跟平常用的普通非事务的代码只有一点点区别了。不理这个,因为我也没有觉得有什么区别3:PROPAGATION_MANDATORY 必须在一个事务中运行。也 就是说, 他只能被一个父事务调用。 否则,他就要抛出异常。4: PROPAGATION_REQUIRES_NEW 这个就比较绕口了。比如我们设计ServiceA.methodA 的事务级别为PROPAGATION_REQUIRED ,ServiceB.methodB 的事务级别 为 PROPA

10、GATION_REQUIRES_NEW ,那么当执行到ServiceB.methodB 的时候, ServiceA.methodA 所在的事务就 会挂起, ServiceB.methodB 会起一个新的事务,等待ServiceB.methodB 的事务完成以后,他才继续执行。他与PROPAGATION_REQUIRED 的事务区别在于事务的回滚程 度了。因为 ServiceB.methodB 是新起一个事务,那么就是存 在两个不同的事务。如果 ServiceB.methodB 已经提交,那么ServiceA.methodA 失败回滚, ServiceB.methodB 是不会回滚 的。如果 S

11、erviceB.methodB 失败回滚,如果他抛出的异常被ServiceA.methodA 捕获, ServiceA.methodA 事务仍然可能提 交。 5: PROPAGATION_NOT_SUPPORTED 当前不支持 事务。比如 ServiceA.methodA 的事务级别是PROPAGATION_REQUIRED ,而 ServiceB.methodB 的事务 级别是 PROPAGATION_NOT_SUPPORTED ,那么当执行到ServiceB.methodB 时, ServiceA.methodA 的事务挂起,而他 以非事务的状态运行完,再继续 ServiceA.metho

12、dA 的事务。6: PROPAGATION_NEVER不能在事务中运行。假设ServiceA.methodA 的事务级别是PROPAGATION_REQUIRED ,而 ServiceB.methodB 的事务级别是 PROPAGATION_NEVER,那么 ServiceB.methodB就要抛出异常了。7: PROPAGATION_NESTED 理解Nested 的关键是save po int。他与PROPAGATIONREQUIRES_NEW的区别是,PROPAGATION_REQUIRES_NEW另起一个事务, 将会与他的父事务相互独立,而 Nested 的事务和他的父事务是相依 的,

13、他的提交是要等和他的父事务一块提交的。也就是说, 如果父事务最后回滚, 他也要回滚的。而Nested事务的好处 是他有一个 savepoint。*ServiceA/savepoint/PROPAGATION_NESTED(SomeException) void methodA() ServiceB.methodB();级别 catch/执行其他业务 , 如tryServiceC.methodC();也就是*说 ServiceB.methodB 失败回滚, 那么 ServiceA.methodA 也会 回滚到save point点上,ServiceA.methodA 可以选择另外一个 分支,比如

14、 ServiceC.methodC ,继续执行,来尝试完成自己的事务。但是这个事务并没有在EJB标准中定义。二、Isolation2、 REPEATABLE READLevel(事务隔离等级):1、Serializable:最严格的级别,事务串 行执行,资源消耗最大; 2、 REPEATABLE READ :保证了个事务不会修改已经由另一个事务读取但未提交(回滚) 的数据。避免了“脏读取”和“不可重复读取”的情况,但是带来了更多的性能损失。3、 READ COMMITTED: 大多数主流数据库的默认事务等级,保证了一个事务不会读到另个并行事务已修改但未提交的数据,避免了“脏读取”。该级别适用于

15、大多数系统。4、Read Uncommitted :保证了读取 过程中不会读取到非法数据。隔离级别在于处理多事务的并 发问题。我们知道并行可以提高数据库的吞吐量和效率,但 是并不是所有的并发事务都可以并发运行,这需要查看数据 库教材的可串行化条件判断了。这里就不阐述。我们首先说并发中可能发生的 3中不讨人喜欢的事情 1:Dirty reads-读脏数据。也就是说,比如事务 A 的未提交(还依然缓存) 的数据被事务 B 读走, 如果事务 A 失败回滚, 会导致事务 B所读取的的数据是错误的。 2:non-repeatable reads-数据不可重复读。比如事务 A中两处读取数据-total-的

16、值。在第读的时候, total 是 100,然后事务 B 就把 total 的数据改成200,事务 A 再读一次,结果就发现, total 竟然就变成 200了,造成事务 A 数据混乱。 3:phantom reads-幻象读数据,这个和non-repeatable reads相似,也是同一个事务中多次读不一致的问题。但是 non-repeatable reads 的不一致是因 为他所要取的数据集被改变了(比如 total 的数据),但是Select account.idphantom reads所要读的数据的不一致却不是他所要读的数据集改变,而是他的条件数据集改变。比如where ="ppgogo*", 第一次读去了 6 个符合条件 的id,第二次读取的时候,由于事务 b把一个帐号的名字由"dd"改成"ppgogol",结果取出来了 7个数据。Dirty readsnon-repeatable readsphantom readsSerializable不会不会不会 REPEATABLEREAD不会不会会 READ COMMITTED不会会 Read Unc

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论