首頁 > 軟體

SpringMVC中事務是否可以加在Controller層的問題

2022-02-26 19:03:03

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

百分百生效! 

以上為個人經驗,希望能給大家一個參考,也希望大家多多支援it145.com。


IT145.com E-mail:sddin#qq.com