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得到該事務即可。

百分百生效! 

以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。

推薦閱讀: