1、问题复现
spring 3.0 + hibernate 3.2
spring mvc使用注解方式;service使用@service注解 事务使用@Transactional
事务配置使用
- <tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true" />
- <tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true" />
在插入或更新数据时,无报错,但数据库中无结果,而查询正常。疑为事务未提交。
2、问题检查
当修改dao层实现逻辑为:
- Assert.notNull(entity, "entity不能为空");
- Transaction ts = getSession().beginTransaction();
- getSession().saveOrUpdate(entity);
- getSession().flush();
- ts.commit();
- logger.debug("save entity: {}", entity);
- Assert.notNull(entity, "entity不能为空");
- Transaction ts = getSession().beginTransaction();
- getSession().saveOrUpdate(entity);
- getSession().flush();
- ts.commit();
- logger.debug("save entity: {}", entity);
- Assert.notNull(entity, "entity不能为空");
- Transaction ts = getSession().beginTransaction();
- getSession().saveOrUpdate(entity);
- getSession().flush();
- ts.commit();
- logger.debug("save entity: {}", entity);
可以正常提交插入、更新。确定为事务未提交。
3、问题分析
spring mvc使用注解方式时需要使用
方式用来扫描该包以及其子包下的@Controller注解的类,纳入spring管理,而同时spring 容器也需要使用这种方式扫描包含@Service、@Components、@Required、@Autowired等注解用来管理bean和完成DI。
- <context:component-scan base-package="com.fengzhiyin" />
- <context:component-scan base-package="com.fengzhiyin" />
出现在spring mvc的配置文件中时,web 容器在扫描包含@Service或@Components的类并包含@Transaction是,此时@Transaction并为完成,导致事务未被注册。
4、问题解决
分两部分扫描:
spring-mvc.xml中扫描controller
application.xml中扫描其他的
mvc 的只扫描controller组件 注意使用 use-default-filters="false"
<context:component-scan base-package="com.fengzhiyin" use-default-filters="false" >
<context:include-filter expression="org.springframework.stereotype.Controller" type="annotation"/>
</context:component-scan>
主体的扫描除controller外的所有组件
<context:component-scan base-package="com.fengzhiyin" >
<context:exclude-filter expression="org.springframework.stereotype.Controller" type="annotation"/>
</context:component-scan>
经调试代码发现:
1、如果不设置use-default-filters="false",则Spring会扫描并优先注册默认的bean(当然包括标记为@Service的bean),这样,标记为@Transactional的service由于transaction manager尚未注册而未能生效,导致事务管理失效。
原理是:标记为@Transactional的service会wrap为经过transactional proxied(不管是CGLIB based或是JDK based)的bean,而不再是纯的service;
2、app的context-scan其实无所谓,但exclude掉controller显然会improve efficiency.