`
Rocky_rup
  • 浏览: 143491 次
  • 性别: 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),并且当黑人儿童获得非常早期的学前教育时,有希望的教育结果是可行的,前提是这些计划包括父母和社区资源参与和参与以及教育程序通过在丰富课程完成后与儿童一起工作的教师...

    node-v10.22.0-darwin-x64.tar.xz

    Node.js,简称Node,是一个开源且跨平台的JavaScript运行时环境,它允许在浏览器外运行JavaScript代码。Node.js于2009年由Ryan Dahl创立,旨在创建高性能的Web服务器和网络应用程序。它基于Google Chrome的V8 JavaScript引擎,可以在Windows、Linux、Unix、Mac OS X等操作系统上运行。 Node.js的特点之一是事件驱动和非阻塞I/O模型,这使得它非常适合处理大量并发连接,从而在构建实时应用程序如在线游戏、聊天应用以及实时通讯服务时表现卓越。此外,Node.js使用了模块化的架构,通过npm(Node package manager,Node包管理器),社区成员可以共享和复用代码,极大地促进了Node.js生态系统的发展和扩张。 Node.js不仅用于服务器端开发。随着技术的发展,它也被用于构建工具链、开发桌面应用程序、物联网设备等。Node.js能够处理文件系统、操作数据库、处理网络请求等,因此,开发者可以用JavaScript编写全栈应用程序,这一点大大提高了开发效率和便捷性。 在实践中,许多大型企业和组织已经采用Node.js作为其Web应用程序的开发平台,如Netflix、PayPal和Walmart等。它们利用Node.js提高了应用性能,简化了开发流程,并且能更快地响应市场需求。

    基于JAVA的物流管理系统的源码设计与实现.zip

    毕业设计物流管理系统的设计与实现(Java版本) 采用Struts2+hibernate+Oracle10g+Tomcat 涉及车辆管理,配送点管理,运输方式管理,订单管理,员工管理,用户管理,部门管理,权限管理,角色管理等基础管理功能。

    基于VB+access实现的成绩分析统计系统(论文+源代码).zip

    基于VB+access实现的成绩分析统计系统(论文+源代码) 【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。

    node-v10.14.2-linux-x64.tar.xz

    Node.js,简称Node,是一个开源且跨平台的JavaScript运行时环境,它允许在浏览器外运行JavaScript代码。Node.js于2009年由Ryan Dahl创立,旨在创建高性能的Web服务器和网络应用程序。它基于Google Chrome的V8 JavaScript引擎,可以在Windows、Linux、Unix、Mac OS X等操作系统上运行。 Node.js的特点之一是事件驱动和非阻塞I/O模型,这使得它非常适合处理大量并发连接,从而在构建实时应用程序如在线游戏、聊天应用以及实时通讯服务时表现卓越。此外,Node.js使用了模块化的架构,通过npm(Node package manager,Node包管理器),社区成员可以共享和复用代码,极大地促进了Node.js生态系统的发展和扩张。 Node.js不仅用于服务器端开发。随着技术的发展,它也被用于构建工具链、开发桌面应用程序、物联网设备等。Node.js能够处理文件系统、操作数据库、处理网络请求等,因此,开发者可以用JavaScript编写全栈应用程序,这一点大大提高了开发效率和便捷性。 在实践中,许多大型企业和组织已经采用Node.js作为其Web应用程序的开发平台,如Netflix、PayPal和Walmart等。它们利用Node.js提高了应用性能,简化了开发流程,并且能更快地响应市场需求。

    ASP+ACCESS网上购物系统设计(源代码+设计说明书+调研报告).zip

    ASP+ACCESS网上购物系统设计(源代码+设计说明书+调研报告).zip

    AO工艺设计计算(全).xls

    污水处理计算书

    node-v7.3.0-x86.msi

    Node.js,简称Node,是一个开源且跨平台的JavaScript运行时环境,它允许在浏览器外运行JavaScript代码。Node.js于2009年由Ryan Dahl创立,旨在创建高性能的Web服务器和网络应用程序。它基于Google Chrome的V8 JavaScript引擎,可以在Windows、Linux、Unix、Mac OS X等操作系统上运行。 Node.js的特点之一是事件驱动和非阻塞I/O模型,这使得它非常适合处理大量并发连接,从而在构建实时应用程序如在线游戏、聊天应用以及实时通讯服务时表现卓越。此外,Node.js使用了模块化的架构,通过npm(Node package manager,Node包管理器),社区成员可以共享和复用代码,极大地促进了Node.js生态系统的发展和扩张。 Node.js不仅用于服务器端开发。随着技术的发展,它也被用于构建工具链、开发桌面应用程序、物联网设备等。Node.js能够处理文件系统、操作数据库、处理网络请求等,因此,开发者可以用JavaScript编写全栈应用程序,这一点大大提高了开发效率和便捷性。 在实践中,许多大型企业和组织已经采用Node.js作为其Web应用程序的开发平台,如Netflix、PayPal和Walmart等。它们利用Node.js提高了应用性能,简化了开发流程,并且能更快地响应市场需求。

    ASP+ACCESS在线考试系统设计(源代码+设计说明书).zip

    ASP+ACCESS在线考试系统设计(源代码+设计说明书).zip

    node-v11.10.0-linux-ppc64le.tar.xz

    Node.js,简称Node,是一个开源且跨平台的JavaScript运行时环境,它允许在浏览器外运行JavaScript代码。Node.js于2009年由Ryan Dahl创立,旨在创建高性能的Web服务器和网络应用程序。它基于Google Chrome的V8 JavaScript引擎,可以在Windows、Linux、Unix、Mac OS X等操作系统上运行。 Node.js的特点之一是事件驱动和非阻塞I/O模型,这使得它非常适合处理大量并发连接,从而在构建实时应用程序如在线游戏、聊天应用以及实时通讯服务时表现卓越。此外,Node.js使用了模块化的架构,通过npm(Node package manager,Node包管理器),社区成员可以共享和复用代码,极大地促进了Node.js生态系统的发展和扩张。 Node.js不仅用于服务器端开发。随着技术的发展,它也被用于构建工具链、开发桌面应用程序、物联网设备等。Node.js能够处理文件系统、操作数据库、处理网络请求等,因此,开发者可以用JavaScript编写全栈应用程序,这一点大大提高了开发效率和便捷性。 在实践中,许多大型企业和组织已经采用Node.js作为其Web应用程序的开发平台,如Netflix、PayPal和Walmart等。它们利用Node.js提高了应用性能,简化了开发流程,并且能更快地响应市场需求。

    毕业设计基于知识图谱和循环神经网络的推荐系统python源码+数据集.zip

    毕业设计基于知识图谱和循环神经网络的推荐系统python源码+数据集.zip毕业设计基于知识图谱和循环神经网络的推荐系统python源码+数据集.zip毕业设计基于知识图谱和循环神经网络的推荐系统python源码+数据集.zip毕业设计基于知识图谱和循环神经网络的推荐系统python源码+数据集.zip 毕业设计基于知识图谱和循环神经网络的推荐系统python源码+数据集.zip 毕业设计基于知识图谱和循环神经网络的推荐系统python源码+数据集.zip毕业设计基于知识图谱和循环神经网络的推荐系统python源码+数据集.zip

    2024年老人机行业分析报告.pptx

    行业报告

    基于matlab实现的导线网平差,主要是附和导线平差程序,用于计算各点坐标并评定其精度 .rar

    基于matlab实现的导线网平差,主要是附和导线平差程序,用于计算各点坐标并评定其精度。.rar

    基于VB+access实现的学生学籍管理系统(系统+论文).zip

    基于VB+access实现的学生学籍管理系统(系统+论文) 【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。

    ASP基于BS的工艺品展示系统的设计与实现(源代码+设计说明书).zip

    ASP基于BS的工艺品展示系统的设计与实现(源代码+设计说明书).zip

    经典SBR设计计算(全).xls

    污水处理计算书

    ASP+ACCESS网上花店毕业设计全套(设计说明书+源代码+说明).zip

    ASP+ACCESS网上花店毕业设计全套(设计说明书+源代码+说明).zip

    污水工艺设计计算书.xlsx

    污水处理计算书

    2024年速冻包子行业分析报告.pptx

    行业报告

Global site tag (gtag.js) - Google Analytics