最近看mybatis的时候做了一个练习,但是进行事务处理的时候出了问题,如下
package com.henu.lz.controller; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Controller; import org.springframework.transaction.annotation.Transactional; import org.springframework.ui.Model; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestParam; import com.henu.lz.dao.PersonMapper; import com.henu.lz.model.Person; @Controller public class SupplierController { @Autowired PersonMapper personMapper; @Transactional @RequestMapping("/add") public String addPerson( @RequestParam("name1") String name1, @RequestParam("age1") int age1, @RequestParam("name2") String name2, @RequestParam("age2") int age2, Model model) { Person person1 = new Person(); Person person2 = new Person(); person1.setName(name1); person1.setAge(age1); person2.setName(name2); person2.setAge(age2); personMapper.save(person1); personMapper.save(person2); model.addAttribute("message", "添加成功!"); return "success"; } }
spring容器和springmvc的配置都没有问题,dao层就是mybatis比较与众不同的的写有sql的xml以及接口。在从前台传值的时候person1正常传,person2传能抛SQLException的值,按理说事务应该回滚的,数据库中不会有person1,但是查看数据库却有person1。用的mysql,引擎设置为innodb后还是这样,换了oracle之后依然如此。这两次save不在一个事务。
网上看了下别人的经历,有的是try…catch之后自行处理没有throw,有的是说设置rollbackFor……有一个比较接近
用的是 hibernate 3.2,在配置dao和controller的时候都用了注解方式自动扫描
这样说是导致dao中事务无效,我试过这样配置,服务启动直接就出错了,很显然冲突,而且用context:include-filter和context:exclude-filter来屏蔽掉controller层的解决办法也有点没必要,分别扫描dao层和controller层不就行了,我是这样配置的。
查看mybatis官网上的project,是比我多了一层service层,然后在service中用的@transactional,我加了一层
package com.henu.lz.service; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import com.henu.lz.dao.PersonMapper; import com.henu.lz.model.Person; @Service(value="personService") public class PersonServiceImpl implements PersonService { @Autowired private PersonMapper personMapper; @Transactional public void save(Person p1, Person p2) { personMapper.save(p1); personMapper.save(p2); } }
相应的,controller也修改为
package com.henu.lz.controller; import javax.annotation.Resource; import org.springframework.stereotype.Controller; import org.springframework.ui.Model; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestParam; import com.henu.lz.model.Person; import com.henu.lz.service.PersonService; @Controller public class SupplierController { @Resource private PersonService personService; @RequestMapping("/add") public String addPerson( @RequestParam("name1") String name1, @RequestParam("age1") int age1, @RequestParam("name2") String name2, @RequestParam("age2") int age2, Model model) { Person person1 = new Person(); Person person2 = new Person(); person1.setName(name1); person1.setAge(age1); person2.setName(name2); person2.setAge(age2); personService.save(person1, person2); model.addAttribute("message", "添加成功!"); return "success"; } }
这时再按开始那样传值能得到预期结果了。
一开始也注意到少了一层service,但是觉得在controller中做同样的事情也可以的,所以就杯具了。
因为spring的context和mvc是分开的,貌似controller不能被注册到spring的context中,于是不能被transactionManager拦截,那么controller中那个@transactional就不起作用了。看了声明式事务的五种配置,都是对注册到context中的bean起作用的,不论是拦截器方式还是aop:config方式。
事务注解方式不能应用于接口,我的mapper用的又是xml方式的,所以只能加一层service,然后controller中的交易就放在service层。