springmvc的controller中使用@Transactional无效

最近看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……有一个比较接近

http://www.iteye.com/topic/714686

用的是 hibernate 3.2在配置dao和controller的时候都用了注解方式自动扫描

<context:component-scan base-package="com..." />

 这样说是导致dao中事务无效,我试过这样配置,服务启动直接就出错了,很显然冲突,而且用context:include-filtercontext: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层。

 

不知道大家有木有遇到过类似现象,有好的建议希望不吝赐教啊

 

你可能感兴趣的:(transactional)