java

关注公众号 jb51net

关闭
首页 > 软件编程 > java > SpringMVC事务加在Controller层

SpringMVC中事务是否可以加在Controller层的问题

作者:手不要乱摸

这篇文章主要介绍了SpringMVC中事务是否可以加在Controller层的问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

SpringMVC中事务是否可以加在Controller层

一般而言,事务都是加在Service层的,但是爱钻牛角尖的我时常想:事务加在Controller层可不可以。

我一直试图证明事务不止可以加在Service层,还可以加在Controller层,但是没有找到有力的论据来支持我这个想法,搞得我一度认为事务只能加在Service层,直到我读过spring官方文档并实践之后,我知道我的想法是对的。

在spring-framework-reference.pdf文档中有这样一段话:

<tx:annotation-driven/> only looks for @Transactional on beans in the same application context it is defined in. This means that, if you put <tx:annotation-driven/> in a WebApplicationContext for a DispatcherServlet, it only checks for @Transactional beans in your controllers, and not your services. 

这句话的意思是,<tx:annoation-driven/>只会查找和它在相同的应用上下文件中定义的bean上面的@Transactional注解,如果你把它放在Dispatcher的应用上下文中,它只检查控制器上的@Transactional注解,而不是你services上的@Transactional注解。

于是,我将事务配置定义在Spring MVC的应用上下文(*-servlet.xml)中,将@Transactional注解打在Controller上,终于事务起作用了。

综上,在Spring MVC中,事务不仅可以加在Service层,同样也可以加在Controller层(虽然不推荐这么做,但至少满足了我的好奇心,(*^__^*) 嘻嘻……)。

记录一下自己的情况,当时是相当于二次开发,什么都是配置好的。但是很坑的是只有一个controller层,当时也没觉得什么,就跟着在controller里面写。结果报错之后发现事务没有回滚,这就很尴尬了。一检查,配置文件里面配置了事务,注解也是写了的,一脸懵。

读了这篇文章后发现,自己的事务是配置在spring的配置文件(一般都是配置在这里),但是我只有controller层,那就得配置到spring-mvc的配置文件里面,换了配置后就好了。

另外,项目是配置了双数据源。这里在记录一下。  

<!-- (事务管理)transaction manager, use JtaTransactionManager for global tx -->
<bean id="transactionManagerMS"class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <property name="dataSource" ref="dataSourceMS" />
</bean>
 
<!-- 可通过注解控制事务 -->
<tx:annotation-driven transaction-manager="transactionManagerMS" />
 
<!-- (事务管理)transaction manager, use JtaTransactionManager for global tx -->
<bean id="transactionManager"class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <property name="dataSource" ref="dataSource" />
</bean>
 
<!-- 可通过注解控制事务 -->
<tx:annotation-driven transaction-manager="transactionManager" />

用的时候通过注解Transactional 就行了。但是一个controller里面涉及到两个数据库的事务的话就只能手动开启事务了 

Spring在Controller层的事务操作

以下是代码

package cn.hr.controller;
import java.io.PrintWriter;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.transaction.PlatformTransactionManager;
import org.springframework.transaction.TransactionDefinition;
import org.springframework.transaction.TransactionStatus;
import org.springframework.transaction.support.DefaultTransactionDefinition;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import cn.hr.base.action.BaseAction;
import cn.hr.service.IAnnualbonuService;
@Controller
@RequestMapping(value="/demoTest")
public class DemoTestController extends BaseAction<Object>{
    @Autowired
    private IAnnualbonuService annualbonuService;
    @Autowired
    private PlatformTransactionManager transactionManager;
    /**
      * 
      * @param payNamelist
      * @param request
      * @param response
      * @param session
      */
    @RequestMapping(value = "/deletePayNamelist",method = RequestMethod.POST)
    public void deletePayNamelist(HttpServletRequest request, HttpServletResponse response,HttpSession session) {
        PrintWriter out = null;
        TransactionStatus status = this.transaction();
        try {
            //=====================业务逻辑处理地方================================
            out=response.getWriter();
            out.write("0");
            out.flush();
            transactionManager.commit(status);
        } catch (Exception e) {
            transactionManager.rollback(status);
            out.write("");
            out.flush();
            logger.error(e);
        }finally{     
            out.flush();
            out.close();
       }
    }
    private TransactionStatus transaction(){
        DefaultTransactionDefinition defaultTransactionDefinition = new DefaultTransactionDefinition();
        defaultTransactionDefinition.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED);
        TransactionStatus status = transactionManager.getTransaction(defaultTransactionDefinition);
        return status;
    }
}

主要是transaction这个方法,意思是:new 一个新的事务,再设置自己所需要的事务隔离级别,最后通过注入的transactionManager得到该事务即可。

百分百生效! 

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

您可能感兴趣的文章:
阅读全文