简述Hibernate 3中的XML持久性
最近发布的Hibernate 3 XML持久性特性,Java开发者现在拥有了一个框架组件,它为易于实现的对象关系(OR)和XML持久性提供了高效的和一致的方法。下面加以说明。
Hibernate的易用性、高性能和对象关系持久性等高级特性给IT界带来了很大的惊喜。Hibernate的最新版本(版本3,3月29日发布的)给产品API带来了一个重要的新特性:Hibernate 3 XML持久性。有了Hibernate 3 之后,Java应用程序开发者可以轻易地把XML文档合并到关系型数据库中。
这个新特性应该明确地告诉已有的Hibernate开发者,因为它也遵循POJO(纯的旧Java对象)相同的一致性方法,需要学习的知识最少。Hibernate 3 XML持久性的优点也应该介绍给新用户。本文讲解的是Hibernate 3持久性方法。
XML持久性为什么重要
大多数大型商业数据库都支持某种形式的本地XML持久性。由于XML持久性是一个相对较新的机制--即使对大型厂商也是如此,这个领域中的标准还在不断地浮现。其结果是,为了把无处不在的关系型持久性机制与日益增长的XML解决方案集成在一起,架构师必须依赖厂商特定的特性或者实现定制的XML持久性框架组件。这两个选择都没有太大的吸引力。厂商特定的特性不是普及的,因为可能产生厂商封锁(lock-in),而定制的框架组件实现可能耗费大量的时间和财力,导致代码难于维护。
在OR(对象关系)持久性方面,Hibernate XML持久性是一个自然而然的解决方案。它可以跨越Hibernate支持的所有关系型平台(如虚拟的或真实的关系型平台)移动,允许自由的迁移对象、基于XML的应用程序和集成解决方案而不用担心下层的关系型实现方法。
体系结构的细节信息
Hibernate是一个良好架构的框架组件,它无缝地利用了本地的环境,不需要用户进行任何特殊的干涉或安装操作。从一个数据库切换到另外一个数据库通常只需要改变驱动程序,并配置Hibernate(在线配置设置信息)来使用另外一种数据库语言。
Hibernate利用dom4j框架组件进行XML的分析和维护。如果需要完全利用Hibernate的XML特性,你就必须对dom4j非常熟悉。一般来说,你会发现dom4j比Java提供的JAXP或与JAXP兼容的XML分析器要容易使用一些。它要求我们学习的相关知识较少,并且利用最少的dom4j知识你就能够高效率地使用Hibernate XML持久性。
实际例子:价格目录同步
通用的电子商务案例可以演示XML关系持久性机制的作用。我们来考虑一个示例,在这个例子中XML集成了在线零售商和供应商之间的产品标价目录。
该电子目录包含了已标价的产品列表。在线商店销售产品,通过自己的存货清单来管理(类似于Amazon与Toys-R-Us和运动产品商店之间的关系)。为了精确地和有效地反映价格的变化,在线零售商必须频繁地接收产品价格信息。它把这些信息存放为XML文档,如下所示:
<products> <product prod_id="3" sku="100101"> <description>Athlete mode body fat scale</description> <list_price>100.00</list_price> <drop_price>60.00</drop_price> </product> <product prod_id="4" sku="100102"> <description>Thermometer</description> <list_price>20.00</list_price> <drop_price>11.00</drop_price> </product> </products>
全面的主要的产品价格列表存储在数据库中,如下所示:
CREATE TABLE PRODUCT ( id INT UNIQUE NOT NULL, description VARCHAR(45) NOT NULL, sku VARCHAR(45) UNIQUE NOT NULL, list_price FLOAT, base_price FLOAT, order_price FLOAT, CONSTRAINT PK_PRODUCT PRIMARY KEY (id ) )
在线零售商通过已有的OR映射提供定价目录的Web表现形式,定价产品都表现为demo.Product Java对象:
/** Product对象表现了定价目录项*/ public class Product { int id; String sku; String description; Double listPrice; Double basePrice; Double orderPrice;
这些对象按照下面的方式映射(为了清楚,我们列出了列名,尽管在属性和列名相匹配的时候Hibernate可以自动地把属性映射为列名):
<?xml version="1.0"?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "替换hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"> <hibernate-mapping package="demo"> <class name="Product" table="product" node="product"> <id name="id" type="int" node="@prod_id" column="id"> </id> <property name="sku" node="@sku" column="sku" not-null="true"/> <property name="description" node="description" column="description" not-null="true"/> <property name="listPrice" node="list_price" column="list_price" /> <property name="basePrice" node="drop_price" column="base_price"/> <property name="orderPrice" column="order_price"/> </class> </hibernate-mapping>
在这种情况下,Hibernate的XML关系持久性就非常方便了。由于该电子商务应用程序接收了包含产品价格更新的XML,它就利用 Hibernate的XML持久性机制把这些XML写入到产品数据库中。Hibernate提供了几种XML持久性选择,包括Hibernate的 saveOrUpdate方法:
document = saxReader.read(inputXML); List users = document.selectNodes("//product"); try { Session session = ibernateUtil.sessionFactory.openSession(); Transaction transaction = session.beginTransaction(); Session dom4jSession = session.openSession(EntityMode.DOM4J); Iterator iter = users.iterator(); while (iter.hasNext()) { Object next = iter.next(); dom4jSession.saveOrUpdate("demo.Product", next ); }// end while transaction.commit(); session.close();
XML映射语法
上面的例子中使用的映射文件不用于Hibernate 2的映射文件。Hibernate 3引入了几种专门用于XML持久性的新映射类型。
主要的新映射属性是节点(node),它与被映射的XML文档中的一个元素或文档中的属性相关联。一个"节点"可能表现为下面的某种映射:・ "element-name(元素名)":在例子中,<product></product>元素会被表示为node="product"。・ "@attribute-name(属性名)":在例子中,node="@sku"会被映射为XML属性<product sku="1001001">。・ ".(句点)":映射为元素的父元素(例如<products>就<product>是的父元素)。・ "element-name/@attribute-name(元素名/属性名)":映射为命名元素的属性(product/@sku)。
XML持久性并非Hibernate的主要任务
Hibernate 3框架组件高效率地实现了目前最通用的一些方法(除了LDAP之外)。Java社团现在拥有了一套框架组件,它为易于实现的OR和XML持久性提供了高效率的和一致性的方法。