問題描述:
由于工作需要,需要在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排除在外
第二種
使用的mysql數據庫,數據庫引擎使用的不是InnoDB,只要切換成InnoDB就可以了
1 alter table tablename type=InnoDb
第三種
注解加的方法權限修飾符不是public,@Transational失效,修改修飾符為public
結果是,我的注解掃描范圍只有controller層,所以第一種方案失敗,數據庫是Oracle數據庫,第二種解決方案失敗,方法權限修飾符就是public,第三種方案失敗......
接下來,繼續查找原因,然后了解到原來事務是和數據源綁定的,如果不給事務管理器qualifer屬性,@Transactional默認會與第一個事務管理器綁定,結果我用的是第二個數據源,導致Transactional失效
解決方案:
配置事務管理器時添加value區分字段
java代碼中添加value
1 @Transactional(value = "ynw")
2 public HashMap<String, Object> addAppointMent(Map map) {}
這樣事務就可以使用了