問題描述: 由於工作需要,需要在spring中配置兩個數據源,有一天突然發現@Transactional註解失效 環境框架: springmvc+spring+spring jdbcTemplate 資料庫: Oracle資料庫 解決方法: 百度獲得三種可能原因及相應解決方案: 第一種 spring ...
問題描述:
由於工作需要,需要在spring中配置兩個數據源,有一天突然發現@Transactional註解失效
環境框架:
springmvc+spring+spring jdbcTemplate
資料庫:
Oracle資料庫
解決方法:
百度獲得三種可能原因及相應解決方案:
第一種
1 <!– 自動掃描的包名 –> 2 <context:component-scan base-package=“com.yxyj” > 3 </context:component-scan>
spring容器優先載入由ServletContextListener(對應applicationContext.xml)產生的父容器,而SpringMVC(對應mvc_dispatcher_servlet.xml)產生的是子容器。子容器Controller進行掃描裝配時裝配的@Service註解的實例是沒有經過事務加強處理,即沒有事務處理能力的Service,而父容器進行初始化的Service是保證事務的增強處理能力的。如果不在子容器中將Service exclude掉,此時得到的將是原樣的無事務處理能力的Service。
springmvc的配置文件里包掃描範圍太大,應該將service排除在外
1 <context:component-scan base-package=“com.will” > 2 <context:exclude-filter type=“annotation” expression=“org.springframework.stereotype.Service” /> 3 </context:component-scan>
第二種
使用的mysql資料庫,資料庫引擎使用的不是InnoDB,只要切換成InnoDB就可以了
1 alter table tablename type=InnoDb
第三種
註解加的方法許可權修飾符不是public,@Transational失效,修改修飾符為public
結果是,我的註解掃描範圍只有controller層,所以第一種方案失敗,資料庫是Oracle資料庫,第二種解決方案失敗,方法許可權修飾符就是public,第三種方案失敗......
接下來,繼續查找原因,然後瞭解到原來事務是和數據源綁定的,如果不給事務管理器qualifer屬性,@Transactional預設會與第一個事務管理器綁定,結果我用的是第二個數據源,導致Transactional失效
解決方案:
配置事務管理器時添加value區分欄位
<bean id="transactionManager02" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> <property name="dataSource" ref="dataSource02" /> <qualifier value="ynw"></qualifier> </bean>
java代碼中添加value
1 @Transactional(value = "ynw") 2 public HashMap<String, Object> addAppointMent(Map map) {}
這樣事務就可以使用了