`
lovnet
  • 浏览: 6705666 次
  • 性别: Icon_minigender_1
  • 来自: 武汉
文章分类
社区版块
存档分类
最新评论

Spring 实现你自己的事务同步

 
阅读更多
在《Spring Transaction 分析事务属性(事务的基本概念、配置)》基础上



将介绍如何实现你自己的事务同步,只要是活动的事务状态发生变化就会收到TransactionSynchronizationManager的回调。

书中的demo:
使用TransactionSynchronizationManager注册了TransactionSynchronization回调,同时MyTransactionSynchronizationAdapter会根据事务的完成状态去调用MySession.beginTransaction()、MySession.commit()或MySession.rollback()方法。


模拟一个session类:
  1. packagecn.partner4java.myptm;
  2. importjava.io.Serializable;
  3. /**
  4. * 模拟一个session类
  5. * @author partner4java
  6. *
  7. */
  8. publicclassMySession {
  9. /** 用来标识一个session */
  10. privateLong sessionId;
  11. publicvoidsave(Serializable entity){
  12. System.out.println(sessionId +":save");
  13. }
  14. publicvoidbeginTransaction(){
  15. System.out.println(sessionId +":beginTransaction");
  16. }
  17. publicvoidcommit(){
  18. System.out.println(sessionId +":commit");
  19. }
  20. publicvoidrollback(){
  21. System.out.println(sessionId +":rollback");
  22. }
  23. publicLong getSessionId() {
  24. returnsessionId;
  25. }
  26. publicvoidsetSessionId(Long sessionId) {
  27. this.sessionId = sessionId;
  28. }
  29. @Override
  30. publicString toString() {
  31. return"MySession [sessionId="+ sessionId +"]";
  32. }
  33. }

简单模拟SessionFactory:
  1. packagecn.partner4java.myptm;
  2. importorg.springframework.transaction.support.TransactionSynchronization;
  3. importorg.springframework.transaction.support.TransactionSynchronizationManager;
  4. /**
  5. * 简单模拟SessionFactory<br/>
  6. * 通判传递的类都为MySessionFactory而不是MySession,通过MySessionFactory获得当前线程的MySession或者开启一个新的MySession
  7. * @author partner4java
  8. *
  9. */
  10. publicclassMySessionFactory {
  11. /**
  12. * 如果当前线程存在MySession,就使用该MySession,否者开启一个新的MySession
  13. * @return
  14. */
  15. publicMySession getSession(){
  16. //传入this,是因为,我们以当前factory类作为键保存的MySession
  17. if(TransactionSynchronizationManager.hasResource(this)){
  18. returngetCurrentSession();
  19. }else{
  20. returnopenSession();
  21. }
  22. }
  23. /**
  24. * 开启一个新MySession
  25. * @return
  26. */
  27. privateMySession openSession() {
  28. MySession mySession =newMySession();
  29. mySession.setSessionId(System.currentTimeMillis());
  30. //注册进当前线程管理一个Synchronization
  31. TransactionSynchronization transactionSynchronization =newMyTransactionSynchronizationAdapter(this);
  32. TransactionSynchronizationManager.registerSynchronization(transactionSynchronization);
  33. //绑定新开启的一个MySession进当前线程事务管理器
  34. TransactionSynchronizationManager.bindResource(this, mySession);
  35. returnmySession;
  36. }
  37. /**
  38. * 获取当前线程的MySession
  39. * @return
  40. */
  41. privateMySession getCurrentSession() {
  42. MySession mySession = (MySession) TransactionSynchronizationManager.getResource(this);
  43. returnmySession;
  44. }
  45. }

核心事务同步适配器:
  1. packagecn.partner4java.myptm;
  2. importorg.springframework.transaction.support.TransactionSynchronizationAdapter;
  3. importorg.springframework.transaction.support.TransactionSynchronizationManager;
  4. /**
  5. * 核心事务同步适配器<br/>
  6. * 当方法上面定义了@Transactional注解,那么当每次状态发生时就会调用本同步适配器
  7. * for transaction synchronization callbacks
  8. * @author partner4java
  9. *
  10. */
  11. publicclassMyTransactionSynchronizationAdapterextends
  12. TransactionSynchronizationAdapter {
  13. privateMySessionFactory mySessionFactory;
  14. publicMyTransactionSynchronizationAdapter(MySessionFactory mySessionFactory) {
  15. this.mySessionFactory = mySessionFactory;
  16. }
  17. @Override
  18. publicvoidbeforeCommit(booleanreadOnly) {
  19. //readOnly标识是否是一个只读线程
  20. if(!readOnly){
  21. MySession mySession = (MySession) TransactionSynchronizationManager.getResource(mySessionFactory);
  22. mySession.beginTransaction();
  23. }
  24. }
  25. @Override
  26. publicvoidafterCompletion(intstatus) {
  27. MySession mySession = (MySession) TransactionSynchronizationManager.getResource(mySessionFactory);
  28. if(STATUS_COMMITTED == status) {
  29. mySession.commit();
  30. }
  31. //当然,你还可以定义回滚方法
  32. }
  33. }

调用起的DAO:
  1. packagecn.partner4java.dao;
  2. publicinterfaceHelloDao {
  3. publicvoidsaveHello();
  4. }

  1. packagecn.partner4java.dao;
  2. importorg.springframework.jdbc.core.support.JdbcDaoSupport;
  3. importorg.springframework.transaction.annotation.Transactional;
  4. importcn.partner4java.myptm.MySessionFactory;
  5. /**
  6. * 一个hello world dao,起到模拟调用自定义事务同步的作用
  7. * @author partner4java
  8. *
  9. */
  10. publicclassHelloDaoImplextendsJdbcDaoSupportimplementsHelloDao {
  11. privateMySessionFactory mySessionFactory;
  12. publicvoidsetMySessionFactory(MySessionFactory mySessionFactory) {
  13. this.mySessionFactory = mySessionFactory;
  14. }
  15. @Transactional
  16. publicvoidsaveHello(){
  17. mySessionFactory.getSession().save(null);
  18. this.getJdbcTemplate().execute("select * from user");
  19. }
  20. }


配置文件:
  1. <?xml version="1.0"encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:aop="http://www.springframework.org/schema/aop"
  4. xmlns:tx="http://www.springframework.org/schema/tx"
  5. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  6. xsi:schemaLocation="
  7. http://www.springframework.org/schema/beans
  8. http://www.springframework.org/schema/beans/spring-beans.xsd
  9. http://www.springframework.org/schema/tx
  10. http://www.springframework.org/schema/tx/spring-tx.xsd
  11. http://www.springframework.org/schema/aop
  12. http://www.springframework.org/schema/aop/spring-aop.xsd">
  13. <bean id="dataSource"class="org.springframework.jdbc.datasource.DriverManagerDataSource"destroy-method="close">
  14. <property name="driverClassName"value="com.mysql.jdbc.Driver"/>
  15. <property name="url"value="jdbc:mysql://localhost:3306/springdb"/>
  16. <property name="username"value="root"/>
  17. <property name="password"value="123456"/>
  18. </bean>
  19. <bean id="transactionManager"class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
  20. <property name="dataSource"ref="dataSource"/>
  21. </bean>
  22. <tx:annotation-driven transaction-manager="transactionManager"/>
  23. <aop:aspectj-autoproxy />
  24. <bean id="mySessionFactory"class="cn.partner4java.myptm.MySessionFactory"/>
  25. <bean id="helloDao"class="cn.partner4java.dao.HelloDaoImpl">
  26. <property name="dataSource"ref="dataSource"/>
  27. <property name="mySessionFactory"ref="mySessionFactory"></property>
  28. </bean>
  29. </beans>


测试:
  1. ApplicationContext ac =newClassPathXmlApplicationContext("/META-INF/spring/myptm.xml");
  2. HelloDao helloDao = (HelloDao) ac.getBean("helloDao");
  3. helloDao.saveHello();
  4. // 后台打印:
  5. // 1322395163008:save
  6. // 1322395163008:beginTransaction
  7. // 1322395163008:commit

总结:有两个核心的Spring类支持了这个功能,TransactionSynchronization接口,TransactionSynchronizationManager类。
TransactionSynchronizationManager负责管理当前线程在资源,资源可以主动绑定到TransactionSynchronizationManager中。
TransactionSynchronization提供了同步调用,当方法上面定义了@Transactional注解,那么当每次状态发生时就会调用本同步适配器。

PlatformTransactionManager的各种实现也是借助了上面这两个类,你可以查阅一下源码。所以,我们自然而然的也可以自己实现一个PlatformTransactionManager,来管理真正的sessionFactory,然后像其他实现一样,交给Spring,然后再给他声明事务。
分享到:
评论

相关推荐

    spring-transaction-synchronization:Spring事务同步示例

    Spring事务同步示例 基于示例项目。

    分布式事务实践 解决数据一致性

    介绍分布式事务的定义、原则和实现原则,介绍使用Spring框架实现分布式事务的几种方式,包括使用JTA、Spring事务同步、链式事务等,并通过实战介绍其实现。除此以外还介绍了一些分布式事务相关的技术,如幂等性、...

    Spring 2.0 开发参考手册

    9.5.1. 理解Spring的声明式事务管理实现 9.5.2. 第一个例子 9.5.3. 回滚 9.5.4. 为不同的bean配置不同的事务语义 9.5.5. &lt;tx:advice/&gt; 有关的设置 9.5.6. 使用 @Transactional 9.5.7. 插入事务操作 9.5.8. ...

    Spring-Reference_zh_CN(Spring中文参考手册)

    9.5.1. 理解Spring的声明式事务管理实现 9.5.2. 第一个例子 9.5.3. 回滚 9.5.4. 为不同的bean配置不同的事务语义 9.5.5. &lt;tx:advice/&gt; 有关的设置 9.5.6. 使用 @Transactional 9.5.6.1. @Transactional 有关的设置 ...

    spring.net中文手册在线版

    14.5.1.理解Spring.NET声明式事务管理的实现 14.5.2.第一个例子 14.5.3.Transaction特性的设置 14.5.4.通过AutoProxyCreator使用声明式事务 14.5.5.通过TransactionProxyFactoryObject使用声明式事务 14.5.6. 通过...

    Spring中文帮助文档

    9.5.1. 理解Spring的声明式事务管理实现 9.5.2. 第一个例子 9.5.3. 回滚 9.5.4. 为不同的bean配置不同的事务语义 9.5.5. &lt;tx:advice/&gt; 有关的设置 9.5.6. 使用 @Transactional 9.5.7. 事务传播 9.5.8. 通知...

    spring-boot mybaits spring security redis整合

    注解redis缓存数据,Spring-session和redis实现分布式session同步(建议按功能模块划分系统)。 6、日志 =========== logback打印日志,业务日志和调试日志分开打印。同时基于时间和文件大小分割日志文件。 9、...

    spring chm文档

    9.5.1. 理解Spring的声明式事务管理实现 9.5.2. 第一个例子 9.5.3. 回滚 9.5.4. 为不同的bean配置不同的事务语义 9.5.5. &lt;tx:advice/&gt; 有关的设置 9.5.6. 使用 @Transactional 9.5.7. 插入事务操作 9.5.8. ...

    Spring API

    9.5.1. 理解Spring的声明式事务管理实现 9.5.2. 第一个例子 9.5.3. 回滚 9.5.4. 为不同的bean配置不同的事务语义 9.5.5. &lt;tx:advice/&gt; 有关的设置 9.5.6. 使用 @Transactional 9.5.7. 事务传播 9.5.8. 通知...

    2-分布式高级篇 _SpringCloud系统_spring_谷粒商城_分布式_分布式高级篇_

    RabbitMQ柔性事务方案、SpringCloud-Gateway网关、Feign远程调用、Sleuth+Zipkin链路追踪系统、Spring Cache缓存、SpringSession跨子域Session同步方案、基于ElasticSearch7全文检索、异步编排与线程池、压力测试...

    使用spring框架整合DBUtils技术,实现用户转账功能

    利用传统事务解决转账线程安全问题,实现同步转账。

    Spring.3.x企业应用开发实战(完整版).part2

    9.3.3 事务同步管理器 9.3.4 事务传播行为 9.4 编程式的事务管理 9.5 使用XML配置声明式事务 9.5.1 一个将被实施事务增强的服务接口 9.5.2 使用原始的 TransactionProxyFactoryBean 9.5.3 基于tx/aop命名空间的配置 ...

    java事务 - 使用注解

    使用注解控制java事务, 类似spring处理,代码干净简洁

    基于SpringJdbc+freemarker实现,具备Mybatis一样的SQL分离和逻辑标签能力的轻量级JAVA持久层框架

    Minidao产生的初衷是为了解决Hibernate项目,在复杂SQL具备Mybatis一样的灵活能力,同时支持事务同步。O/R mapping不用设置xml,零配置便于维护。不需要了解JDBC的知识。SQL语句和java代码的分离 只需接口定义,无需...

    基于Spring Boot-Spring Cloud-Alibaba企业员工职能权限管理平台【毕业设计、课程设计】

    本项目基于Spring Boot+Spring Cloud+Alibaba实现的企业职能权限管理平台。 采用前后端分离的模式,微服务版本前端(基于 RuoYi-Vue); 后端采用Spring Boot、Spring Cloud & Alibaba; 注册中心、配置中心选型Nacos,...

    Spring3.x企业应用开发实战(完整版) part1

    9.3.3 事务同步管理器 9.3.4 事务传播行为 9.4 编程式的事务管理 9.5 使用XML配置声明式事务 9.5.1 一个将被实施事务增强的服务接口 9.5.2 使用原始的 TransactionProxyFactoryBean 9.5.3 基于tx/aop命名空间的配置 ...

    基于Spring BootSpring Cloud & Alibaba的分布式微服务架构权限管理系统同时提供了 Vue

    基于Spring Boot、Spring Cloud & Alibaba的分布式微服务架构权限管理系统,同时提供了 Vue3 的版本。若依是一套全部开源的快速开发平台,毫无保留给个人及企业免费使用。采用前后端分离的模式,微服务版本前端(基于...

    纯spring多database

    纯spring多数据库版本,此版本不支持多数据库之间的数据同步,只支持单数据库的事务。要想支持多数据库之间的数据事务 需要使用atomikos管理。

    微服务架构面试专题系列(MySQL,JVM,并发编程,RabbitMQ消息中间件,Spring)

    MySql的主从实时备份同步的配置,以及原理(从库读主库的binlog),读写分离 Mysql主从同步的实现原理 MySQL索引背后的数据结构及算法原理 摘要数据结构及算法基础 索引的本质 B-Tree和B+Tree B-Tree B+Tree 带有顺序...

    spring-boot mybaits shiro redis整合

    注解redis缓存数据,Spring-session和redis实现分布式session同步(建议按功能模块划分系统)。 6、日志 =========== logback打印日志,业务日志和调试日志分开打印。同时基于时间和文件大小分割日志文件。 9、...

Global site tag (gtag.js) - Google Analytics