在Spring中数据库事务是经过PlatformTransactionManager进行管理的,jdbcTemplate是不能支持事务的,而可以支持事务的是org.springframework.transaction.support.TransactionTemplate模板,它是Spring所提供的事务管理器的模板
•事务的建立、提交和回滚是经过PlatformTransactionManager接口来完成的。
•当事务产生异常时会回滚事务,在默认的实现中全部的异常都会回滚。咱们能够经过配置去修改在某些异常发生时回滚或者不回滚事务。
•当无异常时,会提交事务。java
支持JTA事务,经常使用的是DataSourceTransactionManager,它继承抽象事务管理器AbstractPlatformTransactionManager,而AbstractPlatformTransactionManager又实现了PlatformTransactionManager。这样Spring就能够如同源码中看到的那样使用PlatformTransactionManager接口的方法,建立、提交或者回滚事务了。mysql
MyBatis框架用得最多的事务管理器是DataSourceTransactionManager(org.springframework.jdbc.datasource.DataSourceTransactionManager),所以下面将以此例进行讲解。若是使用的持久框架是Hibernate,那么你就要用到spring-orm包org.springframework.orm.hibernate4.HibernateTransactionManager了。它们大同小异,通常而言咱们在使用时,还会加入XML的事务命名空间。下面配置一个事务管理器spring
<?xml version='1.0' encoding='UTF-8' ?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.0.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.0.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.0.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd"> <!-- 数据库链接池 --> <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource"> <property name="driverClassName" value="com.mysql.cj.jdbc.Driver"/> <property name="url" value="jdbc:mysql://localhost:3306/springmvc?useSSL=false&serverTimezone=Hongkong&characterEncoding=utf-8&autoReconnect=true"/> <property name="username" value="root"/> <property name="password" value="123456"/> <property name="maxActive" value="255"/> <property name="maxIdle" value="5"/> <property name="maxWait" value="10000"/> </bean> <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate"> <property name="dataSource" ref="dataSource"/> </bean> <!-- 配置数据源事务管理器 --> <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> <property name="dataSource" ref="dataSource"/> </bean> </beans>
这里先引入了XML的命名空间,而后定义了数据库链接池,因而使用了DataSourceTransactionManager去定义数据库事务管理器,而且注入了数据库链接池。这样Spring就知道你已经将数据库事务委托给事务管理器transactionManager管理了。在jdbcTemplate源码分析时,笔者就已经指出,数据库资源的产生和释放若是没有委托给数据库管理器,那么就由jdbcTemplate管理,可是此时已经委托给了事务管理器,因此jdbcTemplate的数据库资源和事务已经由事务管理器处理了。sql
在Spring中可使用声明式事务或者编程式事务,现在编程式事务几乎不用了,由于它会产生冗余,代码可读性较差。声明式事务又能够分为XML配置和注解事务,但XML方式也已经不经常使用了,目前主流方法是注解@Transactional。数据库
用Java配置的方式来实现Spring数据库事务,须要在配置类中实现接口TransactionManagementConfigurer的annota-tionDrivenTransactionManager方法。Spring会把annotationDrivenTransactionManager方法返回的事务管理器做为程序中的事务管理器
代码清单:使用Java配置方式实现Spring数据库事物apache
package com.ssm.chapter13.config; import org.apache.commons.dbcp.BasicDataSourceFactory; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; import org.springframework.jdbc.core.JdbcTemplate; import org.springframework.jdbc.datasource.DataSourceTransactionManager; import org.springframework.transaction.PlatformTransactionManager; import org.springframework.transaction.annotation.EnableTransactionManagement; import org.springframework.transaction.annotation.TransactionManagementConfigurer; import javax.sql.DataSource; import java.util.Properties; @Configuration @ComponentScan("com.ssm.chapter13.*") //使用事务驱动管理器 @EnableTransactionManagement public class JavaConfig implements TransactionManagementConfigurer { //数据源 private DataSource dataSource = null; /** * 配置数据源. * @return 数据源. */ @Bean(name = "dataSource") public DataSource initDataSource() { if (dataSource != null) { return dataSource; } Properties props = new Properties(); props.setProperty("driverClassName", "com.mysql.cj.jdbc.Driver"); props.setProperty("url", "jdbc:mysql://localhost:3306/springmvc?useSSL=false&serverTimezone=Hongkong&characterEncoding=utf-8&autoReconnect=true"); props.setProperty("username", "root"); props.setProperty("password", "123456"); props.setProperty("maxActive", "200"); props.setProperty("maxIdle", "20"); props.setProperty("maxWait", "30000"); try { dataSource = BasicDataSourceFactory.createDataSource(props); } catch (Exception e) { e.printStackTrace(); } return dataSource; } /** * 配置jdbcTemplate * @return jdbcTemplate */ @Bean(name = "jdbcTemplate") public JdbcTemplate initjdbcTemplate() { JdbcTemplate jdbcTemplate = new JdbcTemplate(); jdbcTemplate.setDataSource(initDataSource()); return jdbcTemplate; } /** * 实现接口方法,使得返回数据库事务管理器 */ @Override @Bean(name = "transactionManager") public PlatformTransactionManager annotationDrivenTransactionManager() { DataSourceTransactionManager transactionManager = new DataSourceTransactionManager(); //设置事务管理器管理的数据源 transactionManager.setDataSource(initDataSource()); return transactionManager; } }
实现了TransactionManagementConfigurer接口所定义的方法annotation DrivenTransactionManager,而且咱们使用DataSourceTransactionManager去定义数据库事务管理器的实例,而后把数据源设置给它。注意,使用注解@EnableTransactionManagement后,在Spring上下文中使用事务注解@Transactional,Spring就会知道使用这个数据库事务管理器管理事务了。编程
编程式事务以代码的方式管理事务,换句话说,事务将由开发者经过本身的代码来实现,这里须要使用一个事务定义类接口——TransactionDefinition,暂时不进行深刻的介绍,咱们只要使用默认的实现类——DefaultTransactionDefinition就能够了。
代码清单:编程式事务mvc
package com.ssm.chapter13.main; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; import org.springframework.jdbc.core.JdbcTemplate; import org.springframework.transaction.PlatformTransactionManager; import org.springframework.transaction.TransactionDefinition; import org.springframework.transaction.TransactionStatus; import org.springframework.transaction.support.DefaultTransactionDefinition; public class MainTest { public static void main(String[] args) { ApplicationContext ctx = new ClassPathXmlApplicationContext("ssm/chapter13/spring-cfg.xml");//ctx为Spring IoC容器 JdbcTemplate jdbcTemplate = ctx.getBean(JdbcTemplate.class); //事务定义类 TransactionDefinition def = new DefaultTransactionDefinition(); PlatformTransactionManager transactionManager = ctx.getBean(PlatformTransactionManager.class); TransactionStatus status = transactionManager.getTransaction(def); try { //执行SQL语句 jdbcTemplate.update("insert into t_role(role_name, note) " + "values('role_name_transactionManager', 'note_transactionManager')"); //提交事务 transactionManager.commit(status); } catch (Exception ex) { //回滚事务 transactionManager.rollback(status); } } }
从代码中能够看到全部的事务都是由开发者本身进行控制的,因为事务已交由事务管理器管理,因此jdbcTemplate自己的数据库资源已经由事务管理器管理,所以当它执行完insert语句时不会自动提交事务,这个时候须要使用事务管理器的commit方法,回滚事务须要使用rollback方法。
固然这是最简单的使用方式,由于这个方式已经不是主流方式,甚至几乎是不被推荐使用的方式,之因此介绍是由于它的代码流程更为清晰,有助于将来对编程式事务的理解。框架
编程式事务是一种约定型的事务,在大部分状况下,当使用数据库事务时,大部分的场景是在代码中发生了异常时,须要回滚事务,而不发生异常时则是提交事务,从而保证数据库数据的一致性。ide