您好,登錄后才能下訂單哦!
小編今天帶大家了解如何進行Spring聲明性事務常見問題分析,文中知識點介紹的非常詳細。覺得有幫助的朋友可以跟著小編一起瀏覽文章的內容,希望能夠幫助更多想解決這個問題的朋友找到問題的答案,下面跟著小編一起深入學習“如何進行Spring聲明性事務常見問題分析”的知識吧。
聲明性事務是spring一個很重要的功能,可以避免開發陷入繁瑣的事務控制邏輯中。但是可能是用著太方便了很多人對spring事務原理并不清楚,有必要做一番分析。
下邊以攔截器配置方式進行說明。
一、首先看配置文件:
<bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager"> <property name="sessionFactory" ref="sessionFactory" /> </bean> <bean id="matchAllTxInterceptor" class="org.springframework.transaction.interceptor.TransactionInterceptor"> <property name="transactionManager"> <ref bean="transactionManager" /> </property> <property name="transactionAttributes"> <props> <prop key="get*">PROPAGATION_REQUIRED,readOnly,-Exception </prop> <prop key="find*">PROPAGATION_REQUIRED,readOnly,-Exception </prop> <prop key="search*">PROPAGATION_REQUIRED,readOnly,-Exception </prop> <prop key="save*">PROPAGATION_REQUIRED,-Exception </prop> <prop key="modify*">PROPAGATION_REQUIRED,-Exception </prop> <prop key="send*">PROPAGATION_REQUIRED,-Exception </prop> <prop key="revoke*">PROPAGATION_REQUIRED,-Exception </prop> <prop key="del*">PROPAGATION_REQUIRED,-Exception </prop> <prop key="logging*">PROPAGATION_NOT_SUPPORTED,readOnly,-Exception </prop> <prop key="*">PROPAGATION_SUPPORTS,-Exception </prop> </props> </property> </bean> <bean id="autoProxyCreator" class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator"> <property name="interceptorNames"> <list><idref local="matchAllTxInterceptor" /></list> </property> <property name="proxyTargetClass"><value>true</value></property> <property name="beanNames"> <list><value>*Service</value></list> </property> </bean>
配置***步引入AOP代理autoProxyCreator,使用的是spring默認的jdk動態代理BeanNameAutoProxyCreator。
有兩個屬性要介紹一下:
1、攔截范圍beanNames
例子中攔截范圍是*Service,表示IOC容器中以Service結尾的bean,一般配置在spring.xml,serviceContext.xml之類的spring配置文件。
要注意這里不是值src下邊的類。
bean配置信息:
<bean id="menuService" class="cn.ceopen.bss..service.impl.MenuServiceImpl"/>
有圖有真相,下邊是BeanNameAutoProxyCreator 調試信息。
2、截器interceptorNames
interceptorNames定義事務屬性和事務管理器。
配置第二步就是定義事務屬性:事務傳播范圍、事務隔離級別。
事務屬性沒什么好說的,使用spring進行事務管理的都了解,不在這里詳細說了網上有大量資料。
置第三步,指定事務管理器。
這里用的是HibernateTransactionManager,spring提供對常見orm的事務支持。從spring源碼可以看出HibernateTransactionManager.doGetTransaction()同時支持hibernate和jdbc。支持hibernate和jdbc混合事務,不使用jta方式的話有個前提條件:使用同一個數據源,這里所說的同一個數據源,不僅僅指物理上是同一個,在spring配置文件中也要是同一個。我在開發中遇到過這個問題,最早定義了一個數據baseDataSource,hibernate和jdbc都使用此數據源,后來項目要求使用動態數據源就又配了一個數據源dynamicDataSource僅在hibernate下做了改動,未改動jdbc對應配置,出現了事務控制問題。
出錯了事務配置:
<bean id="sessionFactory" class="com.sitechasia.webx.dao.hibernate3.NamedMoudleHbmLocalSessionFactoryBean"> <property name="dataSource" ref="dynamicDataSource" /> <!--與主題無關,省略部分內容--> </bean> <bean id="dynamicDataSource" class="cn.ceopen.bss.pub.base.dao.RoutingDataSource"> <property name="targetDataSources"> <map key-type="java.lang.String"> <entry key="baseDataSource" value-ref="baseDataSource"/> </map> </property> <property name="defaultTargetDataSource" ref="baseDataSource"/> </bean> <bean id="baseDataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource" destroy-method="close"> <!--與主題無關,省略部分內容--> </bean> <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate"> <!--dataSource應該與sessionFactor一致--> <property name="dataSource"><ref bean="baseDataSource"/></property> </bean> <bean id="abstractJdbcDao" abstract="true"> <property name="jdbc" ref="jdbcTemplate" /> </bean>
dao配置文件:
<bean id="actDao" class="cn.ceopen.bss.impl.ActDaoImpl" parent="abstractJdbcDao"/>
dao中同時支持hibernate操作和jdbc操作。
二、事務屬性傳播
先看這樣一個列子:
1、基于jdk動態代理的AOP事務控制,只能針對接口。
在上邊的配置文件中設置的事務屬性對a3()都不起作用,a3()不能單獨設計事務屬性,只能繼承接口方法的事務屬性。
2、類自身事務嵌套
***種情況:
AbcIService abcService; BcdIService bcdService; abcService.a1(); abcService.a2(); bcdService.b1();
這三個方法對應的事務屬性都起作用。
第二種情況:
方法定義
public void a1() { bcdService.b1(); }
調用:
abcService.a1();
結果:
abcService.a1(); bcdService.b1();
這兩個方法對應的事務屬性都起作用。
第三種情況:
方法定義
public void a1() { this.a2(); }
調用:
abcService.a1();
結果:
abcService.a1(); abcService.a2();
a2()對應的事務屬性配置不起作用。
解決辦法:
1)把a2()拆到另一個類中去;
缺點:麻煩,會把相關業務邏輯拆分了
2)調用是不用this.a2(),用abcService.a2();
public void a1() { abcService.a2(); }
缺點:要在類中注入自身引用。
原因分析:
為什么會出現這種情況呢?
我們在調用abcService.a1();時abcService是從IOC容器獲取的,并AbcServiceImpl而是它的動態代理AbcServiceProxy。
示意圖如下,spring不一定是這么實現的但原理一樣。
AbcServiceProxy.a()方法進行了AOP增強,根據配置文件中事務屬性增加了事務控制。
public void a1() { this.a2(); }
this.a2()這里this指的是AbcIServiceImpl并沒用進行AOP增強,所以沒用應用事務屬性,只能繼承a1()的事務屬性。
public void a1() { abcService.a2(); }
abcService則實際是AbcServiceProxy.a2()所以可以應用事務屬性。
所以在類內部進行方法嵌套調用,如果被嵌套的方法a2()需要區別于嵌套方法a1()的事務屬性,需要:1)在接口公開;2)通過代理調用。
感謝大家的閱讀,以上就是“如何進行Spring聲明性事務常見問題分析”的全部內容了,學會的朋友趕緊操作起來吧。相信億速云小編一定會給大家帶來更優質的文章。謝謝大家對億速云網站的支持!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。