TA的每日心情 | 衰 2021-2-2 11:21 |
---|
签到天数: 36 天 [LV.5]常住居民I
|
一、Propagation (事务的传播属性)
4 t3 s' P6 S0 {! i3 o3 I+ ^7 ~; z/ h4 u6 _; d; v4 `3 R( w# V
Propagation : key属性确定代理应该给哪个方法增加事务行为。这样的属性最重要的部份是传播行为。有以下选项可供使用:PROPAGATION_REQUIRED--支持当前事务,如果当前没有事务,就新建一个事务。这是最常见的选择。$ U: N1 e3 d, U2 j1 H
* p1 J; L% J- @, [( HPROPAGATION_SUPPORTS--支持当前事务,如果当前没有事务,就以非事务方式执行。
* v$ k% p5 E% o$ W4 ?# m9 w3 c/ x, }& }
PROPAGATION_MANDATORY--支持当前事务,如果当前没有事务,就抛出异常。3 y+ X8 `* E6 p/ k. I
) N! o: [+ \$ R' b$ uPROPAGATION_REQUIRES_NEW--新建事务,如果当前存在事务,把当前事务挂起。
4 P, D2 N T+ D/ ?. m# G+ G5 z) j
+ e- p M; @% [% O7 xPROPAGATION_NOT_SUPPORTED--以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。: _. i. W! ^! i; _: b# D
* H [" c: r2 m7 T5 C' {) oPROPAGATION_NEVER--以非事务方式执行,如果当前存在事务,则抛出异常。
$ n3 q, ^, q% X
) G2 K2 `, |2 o8 t+ S1: PROPAGATION_REQUIRED
0 T* C, |0 l9 K+ ^1 u
1 F7 H4 e# N4 a ^4 e" M, A* I7 m加入当前正要执行的事务不在另外一个事务里,那么就起一个新的事务+ _ B+ h. C3 @4 ^3 J z
% R1 O H4 ^* x' R- Y5 w, b
比如说,ServiceB.methodB的事务级别定义为PROPAGATION_REQUIRED, 那么由于执行ServiceA.methodA的时候,
( o8 d0 d7 @& U
; }& A( G; t/ I' y, v& eServiceA.methodA已经起了事务,这时调用ServiceB.methodB,ServiceB.methodB看到自己已经运行在ServiceA.methodA
. N2 f" e' u. E# z! Z9 v
9 i" g/ u! C" D5 @的事务内部,就不再起新的事务。而假如ServiceA.methodA运行的时候发现自己没有在事务中,他就会为自己分配一个事务。
2 t1 T8 M- V* O% x
: Z% J) J \9 q$ d4 ^/ K% j这样,在ServiceA.methodA或者在ServiceB.methodB内的任何地方出现异常,事务都会被回滚。即使ServiceB.methodB的事务已经被
0 I" o" N1 c4 }% ]) O' k& E$ m( d
* T% g; S# V! Z$ {; q提交,但是ServiceA.methodA在接下来fail要回滚,ServiceB.methodB也要回滚: J" b% c. E& `2 Y* M6 o3 c0 @
; _( b8 B3 M( s3 l; W! n. i [' g* X# `2: PROPAGATION_SUPPORTS
& B m) r5 l5 _/ T& V
2 |9 V: k! v& j p* K5 P如果当前在事务中,即以事务的形式运行,如果当前不再一个事务中,那么就以非事务的形式运行- O( Y4 s! a7 C2 |6 [- F
% o ~) w: c3 S2 o R4 n+ a
3: PROPAGATION_MANDATORY
& o! Q$ R9 Z6 L6 r
$ [7 ]- e* o0 ?* V* b# B必须在一个事务中运行。也就是说,他只能被一个父事务调用。否则,他就要抛出异常
0 ^) U' s. s/ ] I( c' F, z5 N. M, |, S
4: PROPAGATION_REQUIRES_NEW1 L( f5 w$ g u7 g9 Z3 f" M
/ z9 U. U/ y7 p" B0 O% e& W
这个就比较绕口了。 比如我们设计ServiceA.methodA的事务级别为PROPAGATION_REQUIRED,ServiceB.methodB的事务级别为PROPAGATION_REQUIRES_NEW,: [1 {; C' J4 K$ r
! F+ U) {% ~+ |6 x$ y% p那么当执行到ServiceB.methodB的时候,ServiceA.methodA所在的事务就会挂起,ServiceB.methodB会起一个新的事务,等待ServiceB.methodB的事务完成以后,* K) T* E0 H! K f
; N; R* @6 D& {: _% l# }" z
他才继续执行。他与PROPAGATION_REQUIRED 的事务区别在于事务的回滚程度了。因为ServiceB.methodB是新起一个事务,那么就是存在
5 c* E5 T1 T9 p' a$ }
7 @# `7 ]* r- Q9 \8 M" m: Q两个不同的事务。如果ServiceB.methodB已经提交,那么ServiceA.methodA失败回滚,ServiceB.methodB是不会回滚的。如果ServiceB.methodB失败回滚,- @4 _9 ~ a8 C/ _3 A- x& [
9 [3 Z. \5 w" P- u7 r4 D如果他抛出的异常被ServiceA.methodA捕获,ServiceA.methodA事务仍然可能提交。
& e" X1 a7 `" Z7 q* \3 j5 o7 {0 e9 {1 ~: u4 G* p- f c+ v' r7 z/ ^4 j3 D
5: PROPAGATION_NOT_SUPPORTED
$ J7 \$ j* @2 K% ?; S9 w$ b. Z( u: G$ K+ ~+ U: E/ Y( F
当前不支持事务。比如ServiceA.methodA的事务级别是PROPAGATION_REQUIRED ,而ServiceB.methodB的事务级别是PROPAGATION_NOT_SUPPORTED ,
$ }+ |* V+ h9 l, D$ {3 Z
7 ~1 O% p$ X4 ]1 \, D! @那么当执行到ServiceB.methodB时,ServiceA.methodA的事务挂起,而他以非事务的状态运行完,再继续ServiceA.methodA的事务。9 N: D* h$ S' t0 t8 X1 k/ S( J
* i# C, M: Z0 k0 U6: PROPAGATION_NEVER2 G5 n: I1 o$ _, [9 I7 d/ p
( x3 ^$ A4 w3 l& D# @" X& b# }
不能在事务中运行。假设ServiceA.methodA的事务级别是PROPAGATION_REQUIRED, 而ServiceB.methodB的事务级别是PROPAGATION_NEVER ,
. F+ N k( y% i( y& c) l# b! d. b1 m+ i. N' e' b/ n! |+ S! T
那么ServiceB.methodB就要抛出异常了。& G8 D" v+ }- O9 B% R5 R
: S j0 E1 D5 |0 N I7: PROPAGATION_NESTED6 \; L9 v" H( [
8 N& h$ Z# |: |* ]3 b
理解Nested的关键是savepoint。他与PROPAGATION_REQUIRES_NEW的区别是,PROPAGATION_REQUIRES_NEW另起一个事务,将会与他的父事务相互独立,# u4 {( C" P3 Q% W
5 s8 P7 k: N6 i% w" ^* {5 F而Nested的事务和他的父事务是相依的,他的提交是要等和他的父事务一块提交的。也就是说,如果父事务最后回滚,他也要回滚的。% _8 R* R# V- E8 W2 O# K9 i4 w
) m! S1 R3 s. w) [* _而Nested事务的好处是他有一个savepoint。
$ Y9 t; y+ l9 d6 u
( ?1 _6 ?2 ~8 L7 N*****************************************
M8 y0 l1 q7 s
0 A' M- Q1 y2 T. K/ f* }9 |. ]2 GServiceA {
' h% J6 a" X/ t. \* N/ K# n R8 Y* N: [" e
/**' k' x+ w8 k4 C- K! s" i! M
/ A1 M. h5 B$ p. _" A, h, R/ Q* 事务属性配置为 PROPAGATION_REQUIRED
# \8 f2 X( H3 ]" s# ^2 w, h6 I: I. D0 a7 h3 B1 I ~2 C
*/% P1 U4 d0 ^- j9 q& q L
) ^; e0 q% `! S4 y5 p) \6 s1 Q: ]
void methodA() {
' c, t: E8 O. B! E5 B! p, u- H2 `; q! N$ p2 E
try {0 X7 {5 O1 ?. e/ ~6 ?1 q0 k, W
' j+ e" ~ E) d9 F3 F//savepoint* E" P' I$ [4 A5 W
) D8 p& M. D( {# m
ServiceB.methodB(); //PROPAGATION_NESTED 级别' d: t6 ~. l. @: O
5 N) q5 y! o( _% W5 g
} catch (SomeException) {
# H7 B' W# k& I/ L7 [+ W; e# d7 H$ x2 g4 D) Q4 M
// 执行其他业务, 如 ServiceC.methodC();8 C0 h+ Y( P- [; p7 R E
' W% T4 D* {! i
}
3 x- T, t) d6 j& J* ~2 b& o5 s) B+ j$ Q% t8 `
}
% n( H/ S0 f k4 I" j$ Z- @! i% Q7 u+ b$ p# N
}, R1 w# \8 Z; o) {, F
% O: L) f" N' ^0 _4 z********************************************
" G) u% ?" s6 L0 m$ ?
9 k- p+ a! l3 p! c5 c也就是说ServiceB.methodB失败回滚,那么ServiceA.methodA也会回滚到savepoint点上,ServiceA.methodA可以选择另外一个分支,比如6 R( Y; L1 u+ |+ q G' b* S1 [% m
( y' h1 y) s- N4 n, }
ServiceC.methodC,继续执行,来尝试完成自己的事务。9 u8 k2 w4 U# M {, n4 f
9 S, k% U5 P1 y- P# t- ?
但是这个事务并没有在EJB标准中定义。$ |, O# R3 k3 k
# u( e: m7 g) \: y( A二、Isolation Level(事务隔离等级):
% _8 C; L9 ~, {' {; D3 P# o$ r3 ]& w" w
1、Serializable:最严格的级别,事务串行执行,资源消耗最大;6 ~" c2 r9 R& X, @; q3 Y
( t9 B$ ]5 w) ? W5 w+ d2、REPEATABLE READ:保证了一个事务不会修改已经由另一个事务读取但未提交(回滚)的数据。避免了“脏读取”和“不可重复读取”的情况,但是带来了更多的性能损失。7 e3 E) J; f; Y5 M' @8 o1 j2 z$ X
; E9 C. @2 K, h* p; [
3、READ COMMITTED:大多数主流数据库的默认事务等级,保证了一个事务不会读到另一个并行事务已修改但未提交的数据,避免了“脏读取”。该级别适用于大多数系统。
+ X# p# ]9 U& w; b8 U
- q6 d8 y" e7 W5 ~2 {) K) O5 I4、Read Uncommitted:保证了读取过程中不会读取到非法数据。隔离级别在于处理多事务的并发问题。7 a9 ]6 ^- k) M5 {# Q+ y( \
/ r/ m, u" e1 g! i
我们知道并行可以提高数据库的吞吐量和效率,但是并不是所有的并发事务都可以并发运行,这需要查看数据库教材的可串行化条件判断了。
& M/ u: @5 P0 l1 G2 _/ h { R9 w3 F/ ^0 }2 C9 O6 I& b0 Y
这里就不阐述。
) |5 r# q9 ^+ |: f% N9 K0 {. f7 c, f5 n$ f3 S
我们首先说并发中可能发生的3中不讨人喜欢的事情! X/ C/ L0 D( g3 p- B
, B0 j) V; l( n3 t h1: Dirty reads--读脏数据。也就是说,比如事务A的未提交(还依然缓存)的数据被事务B读走,如果事务A失败回滚,会导致事务B所读取的的数据是错误的。
8 ~2 U: q) j$ g1 [- ~- B: u/ b; d. G% \0 f8 z' W6 q% c: s
2: non-repeatable reads--数据不可重复读。比如事务A中两处读取数据-total-的值。在第一读的时候,total是100,然后事务B就把total的数据改成 200,事务A再读一次,结果就发现,total竟然就变成200了,造成事务A数据混乱。7 E0 Q! ]( c: d0 |- t' c6 l
1 y+ Z o. E; w% Q1 R0 B" X) x
3: phantom reads--幻象读数据,这个和non-repeatable reads相似,也是同一个事务中多次读不一致的问题。但是non-repeatable reads的不一致是因为他所要取的数据集被改变了(比如total的数据),但是phantom reads所要读的数据的不一致却不是他所要读的数据集改变,而是他的条件数据集改变。比如Select account.id where account.name="ppgogo*",第一次读去了6个符合条件的id,第二次读取的时候,由于事务b把一个帐号的名字由"dd"改成"ppgogo1",结果取出来了7个数据。, B. |3 n" Z& g3 J) y+ P+ Q
5 {' K2 Y# Z% P) [
6 i" Y: h( I; n8 t
2 @* R9 |7 ?! `- v Dirty reads non-repeatable reads phantom reads# J2 k5 X2 b( w' R
Serializable 不会 不会 不会
% x5 ^" K3 I; h) Z) F" x. k* @; p8 zREPEATABLE READ 不会 不会 会9 B% D1 O/ D( t; P x# I
READ COMMITTED 不会 会 会- D, H0 f/ Q1 F
Read Uncommitted 会 会 会: c }# n+ e" i
& a9 m9 A+ O$ y }& p- w0 p i
5 y! N( R* H+ l" D9 W4 I
! S" V# i2 V8 B$ z
4 j3 q5 v' w' Q$ l! [$ W/ D0 p
三、readOnly
: G. Y4 Y; ^% Q/ ^9 P: O+ v' q1 i; e5 @) Z9 Y0 ?& ~, R8 E
事务属性中的readOnly标志表示对应的事务应该被最优化为只读事务。1 M; x/ p3 J. S# ], N0 U
1 Z5 s+ Q2 c( g. [/ P
这是一个最优化提示。在一些情况下,一些事务策略能够起到显著的最优化效果,例如在使用Object/Relational映射工具(如:hibernate或TopLink)时避免dirty checking(试图“刷新”)。. s& q6 m* \% @4 S. l
" `# F& ~$ f) n. m4 X# c( ], F四、Timeout
3 U$ U# o( V# g7 R& V% E! B
3 x) z, `+ Z& e$ b) N% y4 N9 E在事务属性中还有定义“timeout”值的选项,指定事务超时为几秒。在JTA中,这将被简单地传递到J2EE服务器的事务协调程序,并据此得到相应的解释
# ?8 u5 x. \$ J9 ^# W* D+ s- l3 F7 v
2 y2 V7 @# R# R: D# `$ ?6 ~4 @: \, Q( t7 F, { J& L- j1 ^+ o
5 Z+ ]! j* w! @: \
E" Z0 I; F6 d+ R" E& @ p
% |/ H' B" j* B7 {1 p8 N总结:8 f) w7 C/ ?/ h9 g
9 _1 i8 u" u, y3 f; u9 x
<aop:config proxy-target-class="true">
( i' E, k Z6 T# W" Q <aop:advisor pointcut="execution(* com.company..*Manager.*(..))"
6 z8 J, C1 z) ~; P1 z D7 B advice-ref="txAdvice" /> Q2 Y$ B3 `2 [; h- C
</aop:config> B: I8 R9 u; V: z) }
0 j/ R& z& S$ y, a o
<tx:advice id="txAdvice"> 9 n( l/ |8 O) ?3 i
<tx:attributes> & L; C$ F" n6 y1 e; _8 [) T$ I8 ^4 h
<tx:method name="save*" isolation="DEFAULT/READ_COMMITTED/READ_UNCOMMITTED/REPEATABLE_READ/SERIALIZABLE]"/> ! H4 l/ X/ b, n; V, o: T+ B
<tx:method name="update*" <tx:method name="update*" propagation="MANDATORY/NESTED/NEVER/NOT_SUPPORTED/REQUIRED/REQUIRES_NEW/SUPPORTS">/>
! s* f; [7 k! ?1 ]& t3 A: f <tx:method name="delete*" /> + d8 Y3 T1 d* L0 \7 Q( I
<tx:method name="find*" read-only="true" />
/ A* ]" J: }% Q# T </tx:attributes>
6 Q; j' t2 _8 U: {: H& m3 V</tx:advice> ) `1 a& b6 c9 t' X2 v6 m
1 ]9 s ~$ j+ g8 W m. T& W& g) u4 h
# j8 _! f, J( \ ~
6 C6 Y+ X+ U& u y isolation设定事务的隔离级别,事务管理器根据它来控制另外一个事务可以看到本事务内的哪些数据。 定义的5个不同的事务隔离级别: DEFAULT:默认的隔离级别,使用数据库默认的事务隔离级别 READ_COMMITTED:保证一个事务修改的数据提交后才能被另外一个事务读取。另外一个事务不能读取该事务未提交的数据。这种事务隔离级别可以避免脏读出现,但是可能会出现不可重复读和幻像读。 READ_UNCOMMITTED:这是事务最低的隔离级别,它充许别外一个事务可以看到这个事务未提交的数据。这种隔离级别会产生脏读,不可重复读和幻像读。 REPEATABLE_READ:这种事务隔离级别可以防止脏读,不可重复读。但是可能出现幻像读。它除了保证一个事务不能读取另一个事务未提交的数据外,还保证了避免不可重复读。 SERIALIZABLE:这是花费最高代价但是最可靠的事务隔离级别。事务被处理为顺序执行。除了防止脏读,不可重复读外,还避免了幻像读。 * Z: `% m* p) {0 }4 R6 C x' {
propagation定义了7个事务传播行为 REQUIRED: 如果存在一个事务,则支持当前事务。如果没有事务则开启一个新的事务。 SUPPORTS: 如果存在一个事务,支持当前事务。如果没有事务,则非事务的执行。但是对于事务同步的事务管理器,SUPPORTS与不使用事务有少许不同。 REQUIRES_NEW 总是开启一个新的事务。如果一个事务已经存在,则将这个存在的事务挂起。 NOT_SUPPORTED 总是非事务地执行,并挂起任何存在的事务。 NEVER 总是非事务地执行,如果存在一个活动事务,则抛出异常 NESTED:如果一个活动的事务存在,则运行在一个嵌套的事务中. 如果没有活动事务, 则按TransactionDefinition.PROPAGATION_REQUIRED 属性执行。
1 c6 n I! M, W4 s# A) y, J
+ d* n& h! D; ?0 z( R嵌套事务一个非常重要的概念就是内层事务依赖于外层事务。外层事务失败时,会回滚内层事务所做的动作。而内层事务操作失败并不会引起外层事务的回滚。 # u3 q/ a. |4 b% u
" m7 `% [6 y5 z5 B. n+ s
原文转自:http://www.cnblogs.com/shitianzeng/articles/2319090.html& J6 `5 o/ o$ x& ~. F
( c- B7 }1 N9 S b* o$ N |
|