在Spring中数据库事务是通过PlatformTransactionManager进行管理的,jdbcTemplate是不能支持事务的,而能够支持事务的是org.springframework.transaction.support.TransactionTemplate模板,它是Spring所提供的事务管理器的模板
  •事务的创建、提交和回滚是通过PlatformTransactionManager接口来完成的。
  •当事务产生异常时会回滚事务,在默认的实现中所有的异常都会回滚。我们可以通过配置去修改在某些异常发生时回滚或者不回滚事务。
  •当无异常时,会提交事务。

  支持JTA事务,常用的是DataSourceTransactionManager,它继承抽象事务管理器AbstractPlatformTransactionManager,而AbstractPlatformTransactionManager又实现了PlatformTransactionManager。这样Spring就可以如同源码中看到的那样使用PlatformTransactionManager接口的方法,创建、提交或者回滚事务了。

配置事务管理器

  MyBatis框架用得最多的事务管理器是DataSourceTransactionManager(org.springframework.jdbc.datasource.DataSourceTransactionManager),因此下面将以此例进行讲解。如果使用的持久框架是Hibernate,那么你就要用到spring-orm包org.springframework.orm.hibernate4.HibernateTransactionManager了。它们大同小异,一般而言我们在使用时,还会加入XML的事务命名空间。下面配置一个事务管理器

  1. <?xml version='1.0' encoding='UTF-8' ?>
  2. <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  3. xmlns:p="http://www.springframework.org/schema/p" xmlns:aop="http://www.springframework.org/schema/aop"
  4. xmlns:tx="http://www.springframework.org/schema/tx" xmlns:context="http://www.springframework.org/schema/context"
  5. xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.0.xsd
  6. http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.0.xsd
  7. http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.0.xsd
  8. http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd">
  9.  
  10. <!-- 数据库连接池 -->
  11. <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource">
  12. <property name="driverClassName" value="com.mysql.cj.jdbc.Driver"/>
  13. <property name="url" value="jdbc:mysql://localhost:3306/springmvc?useSSL=false&amp;serverTimezone=Hongkong&amp;characterEncoding=utf-8&amp;autoReconnect=true"/>
  14. <property name="username" value="root"/>
  15. <property name="password" value="123456"/>
  16. <property name="maxActive" value="255"/>
  17. <property name="maxIdle" value="5"/>
  18. <property name="maxWait" value="10000"/>
  19. </bean>
  20.  
  21. <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
  22. <property name="dataSource" ref="dataSource"/>
  23. </bean>
  24.  
  25. <!-- 配置数据源事务管理器 -->
  26. <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
  27. <property name="dataSource" ref="dataSource"/>
  28. </bean>
  29.  
  30. </beans>

  这里先引入了XML的命名空间,然后定义了数据库连接池,于是使用了DataSourceTransactionManager去定义数据库事务管理器,并且注入了数据库连接池。这样Spring就知道你已经将数据库事务委托给事务管理器transactionManager管理了。在jdbcTemplate源码分析时,笔者就已经指出,数据库资源的产生和释放如果没有委托给数据库管理器,那么就由jdbcTemplate管理,但是此时已经委托给了事务管理器,所以jdbcTemplate的数据库资源和事务已经由事务管理器处理了。

  在Spring中可以使用声明式事务或者编程式事务,如今编程式事务几乎不用了,因为它会产生冗余,代码可读性较差。声明式事务又可以分为XML配置和注解事务,但XML方式也已经不常用了,目前主流方法是注解@Transactional。

用Java配置方式实现Spring数据库事务

  用Java配置的方式来实现Spring数据库事务,需要在配置类中实现接口TransactionManagementConfigurer的annota-tionDrivenTransactionManager方法。Spring会把annotationDrivenTransactionManager方法返回的事务管理器作为程序中的事务管理器
  代码清单:使用Java配置方式实现Spring数据库事物

  1. package com.ssm.chapter13.config;
  2.  
  3. import org.apache.commons.dbcp.BasicDataSourceFactory;
  4. import org.springframework.context.annotation.Bean;
  5. import org.springframework.context.annotation.ComponentScan;
  6. import org.springframework.context.annotation.Configuration;
  7. import org.springframework.jdbc.core.JdbcTemplate;
  8. import org.springframework.jdbc.datasource.DataSourceTransactionManager;
  9. import org.springframework.transaction.PlatformTransactionManager;
  10. import org.springframework.transaction.annotation.EnableTransactionManagement;
  11. import org.springframework.transaction.annotation.TransactionManagementConfigurer;
  12.  
  13. import javax.sql.DataSource;
  14. import java.util.Properties;
  15.  
  16. @Configuration
  17. @ComponentScan("com.ssm.chapter13.*")
  18. //使用事务驱动管理器
  19. @EnableTransactionManagement
  20. public class JavaConfig implements TransactionManagementConfigurer {
  21.  
  22. //数据源
  23. private DataSource dataSource = null;
  24.  
  25. /**
  26. * 配置数据源. * @return 数据源.
  27. */
  28. @Bean(name = "dataSource")
  29. public DataSource initDataSource() {
  30. if (dataSource != null) {
  31. return dataSource;
  32. }
  33. Properties props = new Properties();
  34. props.setProperty("driverClassName", "com.mysql.cj.jdbc.Driver");
  35. props.setProperty("url", "jdbc:mysql://localhost:3306/springmvc?useSSL=false&serverTimezone=Hongkong&characterEncoding=utf-8&autoReconnect=true");
  36. props.setProperty("username", "root");
  37. props.setProperty("password", "123456");
  38. props.setProperty("maxActive", "200");
  39. props.setProperty("maxIdle", "20");
  40. props.setProperty("maxWait", "30000");
  41. try {
  42. dataSource = BasicDataSourceFactory.createDataSource(props);
  43. } catch (Exception e) {
  44. e.printStackTrace();
  45. }
  46. return dataSource;
  47. }
  48.  
  49. /**
  50. * 配置jdbcTemplate * @return jdbcTemplate
  51. */
  52. @Bean(name = "jdbcTemplate")
  53. public JdbcTemplate initjdbcTemplate() {
  54. JdbcTemplate jdbcTemplate = new JdbcTemplate();
  55. jdbcTemplate.setDataSource(initDataSource());
  56. return jdbcTemplate;
  57. }
  58.  
  59. /**
  60. * 实现接口方法,使得返回数据库事务管理器
  61. */
  62. @Override
  63. @Bean(name = "transactionManager")
  64. public PlatformTransactionManager annotationDrivenTransactionManager() {
  65. DataSourceTransactionManager transactionManager = new DataSourceTransactionManager();
  66. //设置事务管理器管理的数据源
  67. transactionManager.setDataSource(initDataSource());
  68. return transactionManager;
  69. }
  70. }

  实现了TransactionManagementConfigurer接口所定义的方法annotation DrivenTransactionManager,并且我们使用DataSourceTransactionManager去定义数据库事务管理器的实例,然后把数据源设置给它。注意,使用注解@EnableTransactionManagement后,在Spring上下文中使用事务注解@Transactional,Spring就会知道使用这个数据库事务管理器管理事务了。

编程式事务

  编程式事务以代码的方式管理事务,换句话说,事务将由开发者通过自己的代码来实现,这里需要使用一个事务定义类接口——TransactionDefinition,暂时不进行深入的介绍,我们只要使用默认的实现类——DefaultTransactionDefinition就可以了。
  代码清单:编程式事务

  1. package com.ssm.chapter13.main;
  2.  
  3. import org.springframework.context.ApplicationContext;
  4. import org.springframework.context.support.ClassPathXmlApplicationContext;
  5. import org.springframework.jdbc.core.JdbcTemplate;
  6. import org.springframework.transaction.PlatformTransactionManager;
  7. import org.springframework.transaction.TransactionDefinition;
  8. import org.springframework.transaction.TransactionStatus;
  9. import org.springframework.transaction.support.DefaultTransactionDefinition;
  10.  
  11. public class MainTest {
  12.  
  13. public static void main(String[] args) {
  14.  
  15. ApplicationContext ctx = new ClassPathXmlApplicationContext("ssm/chapter13/spring-cfg.xml");//ctx为Spring IoC容器
  16.  
  17. JdbcTemplate jdbcTemplate = ctx.getBean(JdbcTemplate.class);
  18. //事务定义类
  19. TransactionDefinition def = new DefaultTransactionDefinition();
  20. PlatformTransactionManager transactionManager = ctx.getBean(PlatformTransactionManager.class);
  21. TransactionStatus status = transactionManager.getTransaction(def);
  22. try {
  23. //执行SQL语句
  24. jdbcTemplate.update("insert into t_role(role_name, note) " + "values('role_name_transactionManager', 'note_transactionManager')");
  25. //提交事务
  26. transactionManager.commit(status);
  27. } catch (Exception ex) {
  28. //回滚事务
  29. transactionManager.rollback(status);
  30. }
  31.  
  32. }
  33.  
  34. }

  从代码中可以看到所有的事务都是由开发者自己进行控制的,由于事务已交由事务管理器管理,所以jdbcTemplate本身的数据库资源已经由事务管理器管理,因此当它执行完insert语句时不会自动提交事务,这个时候需要使用事务管理器的commit方法,回滚事务需要使用rollback方法。
当然这是最简单的使用方式,因为这个方式已经不是主流方式,甚至几乎是不被推荐使用的方式,之所以介绍是因为它的代码流程更为清晰,有助于未来对编程式事务的理解。

  编程式事务是一种约定型的事务,在大部分情况下,当使用数据库事务时,大部分的场景是在代码中发生了异常时,需要回滚事务,而不发生异常时则是提交事务,从而保证数据库数据的一致性。

spring 配置事务管理器的更多相关文章

  1. spring的annotation-driven配置事务管理器详解

    http://blog.sina.com.cn/s/blog_8f61307b0100ynfb.html ——————————————————————————————————————————————— ...

  2. spring简单事务管理器

    事务管理器 <!-- Transaction manager for a single JDBC DataSource -->  <bean id="transaction ...

  3. ssh框架配置事务管理器

    http://blog.163.com/zsq303288862@126/blog/static/9374596120111182446727/

  4. 解决在Spring整合Hibernate配置tx事务管理器出现错误的问题

    问题描述: Error occured processing XML 'org/aopalliance/intercept/MethodInterceptor'. See Error Log for ...

  5. 跟我学Spring3(9.2):Spring的事务之事务管理器

    原文出处: 张开涛9.2.1 概述 Spring框架支持事务管理的核心是事务管理器抽象,对于不同的数据访问框架(如Hibernate)通过实现策略接口PlatformTransactionManage ...

  6. spring事务管理器的源码和理解

    原文出处: xieyu_zy 以前说了大多的原理,今天来说下spring的事务管理器的实现过程,顺带源码干货带上. 其实这个文章唯一的就是带着看看代码,但是前提你要懂得动态代理以及字节码增强方面的知识 ...

  7. spring的事务管理配置

    spring有两种事务配置器,可以使用spring的jdbc事务管理器,也可以使用对hibernate的事务管理器 第一种 使用Spring JDBC或IBatis进行事务配置(配置文件方式): &l ...

  8. Spring jdbctemplate和事务管理器

    内部bean 对象结构: @Autowiredprivate IAccountService accountService; @Service("accountService")@ ...

  9. spring事务管理器设计思想(二)

    上文见<spring事务管理器设计思想(一)> 对于第二个问题,涉及到事务的传播级别,定义如下: PROPAGATION_REQUIRED-- 如果当前没有事务,就新建一个事务.这是最常见 ...

随机推荐

  1. 关于 or 判断都是Ture的问题

    问题: 1. ret1 = [1, 2, 3, 4] if 11 or 33 in ret1: print("ok") else: print("no") 2. ...

  2. 洛谷 P1279 字串距离 题解

    每日一题 day24 打卡 Analysis 字符串+dp 仔细观察发现,对于f[i][j],它的值为以下三个值中的最小者: f[i-1][j]+k //a[i]对应空格 f[i][j-1]+k // ...

  3. Oracle substr() 字符截取函数

    1.substr函数格式   (俗称:字符截取函数) 格式1: substr(string string, int a, int b); 格式2:substr(string string, int a ...

  4. Ubuntu下彻底卸载默认安装的mysql,自己手动下载安装MYSQL

    彻底卸载: sudo apt-get autoremove --purge mysql-server-5.7 sudo apt-get remove mysql-common sudo rm -rf ...

  5. 【概率论】5-7:Gama分布(The Gamma Distributions Part II)

    title: [概率论]5-7:Gama分布(The Gamma Distributions Part II) categories: - Mathematic - Probability keywo ...

  6. 原创:从海量数据中查找出前k个最小或最大值的算法(java)

    现在有这么一道题目:要求从多个的数据中查找出前K个最小或最大值 分析:有多种方案可以实现.一.最容易想到的是先对数据快速排序,然后输出前k个数字.   二.先定义容量为k的数组,从源数据中取出前k个填 ...

  7. note_4.10

    单位根反演 \[ \frac{1}{k}\sum_{i=0}^{k-1}\omega_k^{in}=[k|n] \] 所以 \[ \begin{equation} \begin{split} \sum ...

  8. SpringBoot整合ElasticSearch:基于SpringDataElasticSearch

    0.注意事项 SpringDataElasticSearch可能和远程的ElasticSearch版本不匹配,会宝座 版本适配说明:https://github.com/spring-projects ...

  9. 美团Android自动化之旅—适配渠道包

    http://tech.meituan.com/mt-apk-adaptation.html 概述 前一篇文章(美团Android自动化之旅-生成渠道包)介绍了Android中几种生成渠道包的方式,基 ...

  10. vue+vue-resource设置请求头(带上token)

    前言 有这样的一个需求,后台服务器要求把token放在请求头里面 嗯一般是通过data里面通过参数带过去的 第一种方法 全局改变: Vue.http.headers.common['token'] = ...