`
Rocky_rup
  • 浏览: 143421 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

解读《使用Jencks实现Hibernate与Jackrabbit的分布式事务》

阅读更多

前言

 

本文是针对《使用Jencks实现Hibernate与Jackrabbit的分布式事务》(以下简称《JHJ》)一文的理论分析。若只关心实现,请忽略此文点击这里跳转到实现的参考示例。

 

本文先简单介绍JTA和JCA的一些概念作为理论基础,之后再分析Jencks如何实现Hibernate与Jackrabbit的分布式事务的。

有关JTA

JTA全称为Java Transaction API,顾名思义JTA定义了一组统一的事务编程的接口,这些接口如下:

 

XAResource

 

XAResource接口是对实现了X/Open CAE规范的资源管理器 (Resource Manager,数据库就是典型的资源管理器) 的抽象,它由资源适配器 (Resource Apdater) 提供实现。XAResource是支持事务控制的核心

Transaction

Transaction接口是一个事务实例的抽象,通过它可以控制事务内多个资源的提交或者回滚。二阶段提交过程也是由Transaction接口的实现者来完成的。

 

TransactionManager

 

托管模式 (managed mode) 下,TransactionManager接口是被应用服务器调用,以控制事务的边界的。

 

UserTransaction

非托管模式 (non-managed mode) 下,应用程序可以通过UserTransaction接口控制事务的边界

 

托管模式下的事务提交场景



 

 

注意 
在上图中3和5的步骤之间省略了应用程序对资源的操作 (如CRUD)。另外,应用服务器什么时机 enlistResource,又是什么时候delistResource呢?这在后文中会解释。

 

 

 

更多细节请参见:

 

  1. 事务服务浅析(第二部分);
  2. Sun Microsystems Inc.Java Transaction API (JTA) Specification Version 1.1。

有关JCA

 


 上图为JCA的架构图,中间涉及元素说明如下:

 

Enterprise Information System

简称EIS,在JTA中它又被称为资源管理器。典型的EIS有数据库,事务处理系统(Transaction Processing System),ERP系统。

 

Resource Adapter

资源适配器(Resource Adaper)是JCA的关键。要想把不同的EIS整合(或者连接)到J2EE运行环境中,就必须为每个EIS提供资源适配器,它会将将EIS适配为一个具备统一编程接口的资源 (Resource) 。这个统一编程接口就是上图中的System ContractsClient API。下面的UML类图将完美诠释资源适配器。

 



 

Application Server

应用服务器 (Application Server) 通过System Contracts来管理对EIS的安全、事务、连接等。典型的应用服务器有JBoss、JOnAS、Geronimo、GlassFish等。

 

Application Component 

应用组件 (Application Component) ,它封装了应用业务逻辑,像对资源的访问和修改。典型的应用组件就是EJB。

 

更多细节请参见:

  1. Sun Microsystems Inc.J2EE Connector Architecture 1.5

 

实现分析

回到用《JHJ》的问题上来,上面关于JTA与JCA到底能够提供哪些帮助呢?总结一下有两点:

 

  1. 我们需要两个资源适配器能够分别将Hibernate(实质是数据库)和Jackrabbit进行适配,以便将XAResource绑定到事务管理器中;
  2. 我们需要一个事务管理器的实现来事务的托管。

事务管理器

众所周知,应用服务器是提供事务管理器实现的,但这不意味着我们只能选择应用服务器,不然就没有必要写《JHJ》和此文了。这里我选择了Jencks,它是一个轻量级的JCA容器,能够很容易与Spring进行集成,并由Spring的JtaTransactionManager将事务管理的职责委派给Jencks。

 

 

提示 
Jencks其实是将连接管理和事务管理都委托给Geronimo来实现的,所以部署Jencks会依赖Geronimo的包。

 

 

资源适配器

 

 

前面提到了资源适配器实现事务支持的关键——XAResource,但它并非直接暴露出来的,需要通过ManagedConnection接口获取,而ManageConnection又由ManageConnectionFactory接口来提供。因此,资源适配器的问题就落实到寻找ManageConnectionFactory接口实现的提供者。

 

Jackrabbit项目中有个组件叫jackrabbit-jca,提供了ManageConnectionFactory接口的实现类JCAManagedConnectionFactory。

数据库方面,Jencks为其提供了ManageConnectionFactory接口的实现类DataSourceMCF。

 

注意 
数据库的驱动必须支持XA,否则数据库是不能参与到分布式事务当中的。

 

 

连接管理器

 

有了事务管理器和资源适配器还没有完,因为在 有关JTA 中有两个问题没有解决:

 

  1. 什么时候向事务绑定资源,即调用Transaction.enlistResource;
  2. 什么时候把资源与事务解绑,即调用Transaction.delistResource;

 

这两个问题由连接管理器解决,连接管理器负责管理ManageConnectionFactory,当应用获取连接时,连接管理器要做两件事:

 

  1. 向ManageConnection注册ConnectionEventListener,以监听Connection关闭的事件,在关闭时delistResource;
  2. 获取当前事务enlistResource。

 

下面结合《JHJ》示例来看连接管理器是如何做到的吧:

 

  • ConnectionFactoryBean将ManageConnectionFactory和ConnectionManager关联上;
<!-- Jackrabbit -->
<bean id="repository"
	class="org.springframework.jca.support.LocalConnectionFactoryBean">
	<property name="managedConnectionFactory">
		<ref local="repositoryManagedConnectionFactory" />
	</property>
	<property name="connectionManager">
		<bean parent="connectionManager" />
	</property>
</bean>

<!-- Database -->
<bean id="dataSource" class="org.jencks.factory.ConnectionFactoryFactoryBean">
	<property name="managedConnectionFactory" ref="jdbcManagedConnectionFactory" />
	<property name="connectionManager">
		<bean parent="connectionManager" />
	</property>
</bean>
  •  ConnectionManagerFactoryBean将ConnectionManager与TransactionManager关联上;
<!-- 链接管理器 -->
<bean id="connectionManager" class="org.jencks.factory.ConnectionManagerFactoryBean" abstract="true">
	<property name="transactionManager">
		<ref local="delegateTransactionManager" />
	</property>
	<property name="transaction" value="xa" />
</bean>

  •  ManagerConnectionFactory创建的连接工厂都注入了ConnectionManager,为是让ConnectionManager来管理连接的分配,并enlistReource。
// DataSourceMCF.java
public Object createConnectionFactory(ConnectionManager connectionManager) 
		throws ResourceException {
	return new DataSource(this, connectionManager);
}

// JCAManagedConnectionFactory.java
public Object createConnectionFactory(ConnectionManager cm)
		throws ResourceException {
	createRepository();
	JCARepositoryHandle handle = new JCARepositoryHandle(this, cm);
	log("Created repository handle (" + handle + ")");
	return handle;
}

 

 

// DataSource.java
public Connection getConnection() throws SQLException {
	try {
		return (Connection) cm.allocateConnection(mcf, containerRequestInfo);
	} catch (ResourceException e) {
		...
	}
}

// JCARepositoryHandle.java	
private Session login(JCAConnectionRequestInfo cri)
		throws LoginException, NoSuchWorkspaceException, RepositoryException {
	try {
		return (Session) cm.allocateConnection(mcf, cri);
	} catch (ResourceException e) {
		...
	}
}

 

 

 

// GenericConnectionManager.java
public Object allocateConnection(ManagedConnectionFactory managedConnectionFactory,
								 ConnectionRequestInfo connectionRequestInfo)
		throws ResourceException {
	ManagedConnectionInfo mci = new ManagedConnectionInfo(managedConnectionFactory, connectionRequestInfo);
	ConnectionInfo ci = new ConnectionInfo(mci);
	getStack().getConnection(ci); // 这里通过拦截器机制完成事件监听注册和enlistReource
	Object connection = ci.getConnectionProxy();
	if (connection == null) {
		connection = ci.getConnectionHandle();
	}
	return connection;
}	

// MCFConnectionInterceptor.java
public void getConnection(ConnectionInfo connectionInfo) throws ResourceException {
	// ManagedConnectionInfo mci = ...
	
	try {
		// ManagedConnection mc = ...
		...
		GeronimoConnectionEventListener listener = new GeronimoConnectionEventListener(stack, mci);
		mci.setConnectionEventListener(listener);
		mc.addConnectionEventListener(listener);
	} catch (ResourceException re) {
		...
	}
}

// TransactionEnlistingInterceptor.java
public void getConnection(ConnectionInfo connectionInfo) throws ResourceException {
	next.getConnection(connectionInfo);
	try {
		ManagedConnectionInfo mci = connectionInfo.getManagedConnectionInfo();

		Transaction transaction = TxUtil.getTransactionIfActive(transactionManager);
		if (transaction != null) {
			XAResource xares = mci.getXAResource();
			transaction.enlistResource(xares);
		}
	} catch (Exception e) {
		...
	}
}

 

 

小结

解读《JHJ》的目的只为抛砖引玉,激发思考和探讨。此外,这里留下一个问题,在Spring的JtaTransactionManager的源码中发现,spring使用的是UserTransaction而不是TransactionManager这是为何呢?

 

 

 

 

  • 大小: 27.2 KB
  • 大小: 38.3 KB
  • 大小: 50.2 KB
5
0
分享到:
评论
1 楼 whaosoft 2009-01-17  
呵呵 学习一下哈
JCA
Jencks
Jackrabbit 这个3个我都不会呵呵

相关推荐

    家庭的社会学习理论:分析

    Jencks,1972;Jensen,1973;Moos &amp; Insel,1974;Schulman,1970;Walberg,1971 )。 环境研究综述(例如,Bloom, 1964; Dave, 1963; Marjoribanks, 1972a; Mosychuk, 1969; Plowden, 1967; Vernon

    正常和学习障碍青少年的配对表现

    一些调查可能过于悲观(Jencks,1972),并且当黑人儿童获得非常早期的学前教育时,有希望的教育结果是可行的,前提是这些计划包括父母和社区资源参与和参与以及教育程序通过在丰富课程完成后与儿童一起工作的教师...

    实验-三、数据库安全性(目的、要求和模板).doc

    实验-三、数据库安全性(目的、要求和模板).doc

    基于Docker搭建K8s集群离线包

    基于Docker搭建K8s集群离线包,包含部署时所需的全部文件,可在内网环境中使用,K8s为1.23.0版本,docker为20.10.9-3版本

    基于springboot+vue实现的求职招聘类型网站源代码+数据库(优质毕设项目).zip

    基于springboot+vue实现的求职招聘类型网站源代码+数据库(优质毕设项目).zip个人经导师指导并认可通过的98分毕业设计项目,主要针对计算机相关专业的正在做毕设的学生和需要项目实战练习的学习者。也可作为课程设计、期末大作业。包含全部项目源码、该项目可以直接作为毕设使用。项目都经过严格调试,确保可以运行! 基于springboot+vue实现的求职招聘类型网站源代码+数据库(优质毕设项目).zip个人经导师指导并认可通过的98分毕业设计项目,主要针对计算机相关专业的正在做毕设的学生和需要项目实战练习的学习者。也可作为课程设计、期末大作业。包含全部项目源码、该项目可以直接作为毕设使用。项目都经过严格调试,确保可以运行! 基于springboot+vue实现的求职招聘类型网站源代码+数据库(优质毕设项目).zip个人经导师指导并认可通过的98分毕业设计项目,主要针对计算机相关专业的正在做毕设的学生和需要项目实战练习的学习者。也可作为课程设计、期末大作业。包含全部项目源码、该项目可以直接作为毕设使用。项目都经过严格调试,确保可以运行!基于springboot+vue实现的求

    基于Android系统Api封装常用工具类.zip

    基于Android系统Api封装常用工具类.zip

    基于PCA和SVM的人脸识别

    svm 基于PCA(主成分分析)和SVM(支持向量机)的人脸识别是一种常见的方法。这里是一个简要说明: PCA(主成分分析): PCA是一种降维技术,它通过线性变换将高维数据转换为低维数据,同时保留最大的数据方差。 在人脸识别中,PCA被用来提取人脸图像中的主要特征,从而减少数据的维度,并保留最重要的信息。 SVM(支持向量机): SVM是一种监督学习算法,用于分类和回归分析。 在人脸识别中,SVM被用来构建一个分类器,以将提取的人脸特征映射到相应的人脸身份标签。 基于PCA和SVM的人脸识别流程: 训练阶段: 收集训练数据集,包括多个人的人脸图像和相应的标签。 对每个人脸图像应用PCA,将其转换为低维特征向量。 使用这些特征向量训练一个SVM分类器,使其能够将人脸特征向量与相应的人脸标签关联起来。 测试阶段: 对待识别的人脸图像应用相同的PCA转换,将其转换为与训练数据相同的低维特征向量。 使用训练好的SVM分类器,将待识别的人脸特征向量与已知的人脸标签进行比较,从而确定其身份。 优点: PCA可以有效地降低数据的维度,减少计算复杂度,并提取最相关的特征。 SVM在处理

    天津科技大学-答辩通用PPT模板我给母校送模板作品.pptx

    PPT模板,答辩PPT模板,毕业答辩,学术汇报,母校模板,我给母校送模板作品,周会汇报,开题答辩,教育主题模板下载。PPT素材下载。

    Java SE Development Kit 11.0.23 macOS x64 DMG Installer

    Java SE Development Kit 11.0.23 macOS x64 DMG Installer

    课设&大作业-SSM毕业设计项目.zip

    【资源说明】【毕业设计】 1、该资源内项目代码都是经过测试运行成功,功能正常的情况下才上传的,请放心下载使用。 2、适用人群:主要针对计算机相关专业(如计科、信息安全、数据科学与大数据技术、人工智能、通信、物联网、数学、电子信息等)的同学或企业员工下载使用,具有较高的学习借鉴价值。 3、不仅适合小白学习实战练习,也可作为大作业、课程设计、毕设项目、初期项目立项演示等,欢迎下载,互相学习,共同进步!

    studyopencv2

    studyopencv2

    实验五-使用matlab实现卷积的运算.pdf

    实验五-使用matlab实现卷积的运算.pdf

    2024年中国纸杯蛋糕盒行业研究报告.docx

    2024年中国纸杯蛋糕盒行业研究报告

    总结作图常用的操作 Python数据分析及可视化-Matplotlib极简入门.zip

    matplotlib绘图 通过 Matplotlib,开发者可以仅需要几行代码,便可以生成绘图、直方图、功率谱、条形图、错误图、散点图等。 Matplotlib基础知识 1.Matplotlib中的基本图表包括的元素 x轴和y轴 水平和垂直的轴线 x轴和y轴刻度 刻度标示坐标轴的分隔,包括最小刻度和最大刻度 x轴和y轴刻度标签 表示特定坐标轴的值 绘图区域 实际绘图的区域 2.hold属性 hold属性默认为True,允许在一幅图中绘制多个曲线;将hold属性修改为False,每一个plot都会覆盖前面的plot。 但是不推荐去动hold这个属性,这种做法(会有警告)。因此使用默认设置即可。 3.网格线 grid方法 使用grid方法为图添加网格线 设置grid参数(参数与plot函数相同) .lw代表linewidth,线的粗细 .alpha表示线的明暗程度 4.axis方法 如果axis方法没有任何参数,则返回当前坐标轴的上下限 5.xlim方法和ylim方法 除了plt.axis方法,还可以通过xlim,ylim方法设置坐标轴范围

    java本科毕业设计基于RFID技术的国有资产管理系统源码后台项目.zip

    高分设计源码,详情请查看资源内容中使用说明 高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明

    springboot部署系统(自动化部署+缓存管理+业务降级+应用监控).zip

    springboot框架 一、Spring Boot基础应用 Spring Boot特征 概念: 约定优于配置,简单来说就是你所期待的配置与约定的配置一致,那么就可以不做任何配置,约定不符合期待时才需要对约定进行替换配置。 特征: 1. SpringBoot Starter:他将常用的依赖分组进行了整合,将其合并到一个依赖中,这样就可以一次性添加到项目的Maven或Gradle构建中。 2,使编码变得简单,SpringBoot采用 JavaConfig的方式对Spring进行配置,并且提供了大量的注解,极大的提高了工作效率,比如@Configuration和@bean注解结合,基于@Configuration完成类扫描,基于@bean注解把返回值注入IOC容器。 3.自动配置:SpringBoot的自动配置特性利用了Spring对条件化配置的支持,合理地推测应用所需的bean并自动化配置他们。 4.使部署变得简单,SpringBoot内置了三种Servlet容器,Tomcat,Jetty,undertow.我们只需要一个Java的运行环境就可以跑SpringBoot的项目了

    CUDA规约求和.cu

    CUDA规约求和.cu

    工程有限公司计算机信息系统数据信息安全管理办法.doc

    工程有限公司计算机信息系统数据信息安全管理办法.doc

    小学教育信息化工作计划.doc

    小学教育信息化工作计划.doc

    springboot停车位管理源码.rar

    springboot停车位管理源码.rarspringboot停车位管理源码.rarspringboot停车位管理源码.rar

Global site tag (gtag.js) - Google Analytics