携手创作,共同成长!这是我参与「掘金日新计划 · 8 月更文挑战」的第20天,点击查看活动详情
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.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.base.action.BaseAction;
import cn.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得到该事务即可。
在SpringBoot事务的使用
1.编程式事务管理
spring框架提供了两种编程式事务管理方式:使用TransactionTemplate和直接使用PlatformTransactionManager实现
2.声明式事务管理
声明式事务有两种方式,一种是在配置文件中做相关的事务规则声明,另一种是基于@Transactional 注解的方式。
二、两种方式的优缺点
1.声明式事务管理是非侵袭入式的开发方式,使业务代码逻辑不受污染。声明式事务管理不需要通过编程的方式管理事务,这样就不需要在业务代码中掺杂事务管理的代码,只需要在配置文件中做相关的事务规则声明或者通过@Transactional注解的方式,便可以将事务规则应用到业务逻辑中。
2.声明式事务管理的作用粒度只能作用到方法级别,无法做到像编程式事务那样可以作用到代码块级别。但是声明式事务管理也可以曲线救国,可以将需要进行事务管理的代码块写成独立的方法。
3.声明式事务管理无法获取事务状态,但是编程式事务管理可以。
所以在实际开发中要结合具体情况选择使用适合自己项目的事务管理方式。