本文共 6900 字,大约阅读时间需要 23 分钟。
A Book business object. @author Duke The unique ISBN code for this book.使用此映射文件,hbm2java会生成看起来象这样的类:
/** * A Book business object. * @author Duke */ public class Book { private String id; private String name; private String isbn; public Book() { } public String getId() { return id; } private void setId(String id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } /** * The unique ISBN code for this book. */ public String getIsbn() { return isbn; } public void setIsbn(String isbn) { this.isbn = isbn; } }为实际项目生成类 实际上,hbm2java被设计成用来把Hibernate映射文件转化成一组相应的Java类。如果你想在实际的应用程序中使用这种方法,很显然,对 于所有的Hibernate映射文件一次性生成类会更为方便。最好的方法是把类生成结合进你的自动化构建过程中。结合进Ant构建过程中使用Ant调用 hbm2java是相当直接的。首先,你需要申明hbm2java任务(task)以便Ant可以调用它:
classname="net.sf.hibernate.tool.hbm2java.Hbm2JavaTask" classpathref="project.class.path"/>接着,你要用到这个任务。例如,通过写一个目标(target)将源目录中的所有*.hbm.xml文件生成源代码。假设$ {src.hibernate}表示含有Hibernate映射文件的目录,${src.generated}就是你想放源代码的地方。这样Ant任务看 起来就象是:
description="Generate Java source code from the Hibernate mapping files">用Maven 1 定制构建过程 为结合进Maven ( 1.0)构建过程,你需要修改maven.xml文件。Maven代码就存放在这个文件中。脚本(script)主要检查了Hibernate映射 文件自上次类生成后是否已被更改(使用uptodate 标记),如果没有,就调用此前所描述的Ant中的hbm2java任务。这种情况下,我们做了以下的假设: ? hbm2java.xml配置文件应在src/hibernate目录中 ? Hibernate映射文件应在src/hibernate目录中 ? 在src/generated/src/java目录下生成Java类
targetfile="${maven.src.dir}/generated/hbm.jar"> value="${hibernateBuild.uptodate}"/> classname="net.sf.hibernate.tool.hbm2java.Hbm2JavaTask" classpathref="maven.dependency.classpath"/> output="${maven.src.dir}/generated/src/java" >用Maven 2 定制构建过程 若碰巧你正使用Maven 2,事情就更简单一点了。把maven-antrun-plugin插件添加到pom.xml文件中,而不是在maven.xml文件中使用完整的 goals属性(pre and post goals)。在此插件中的task那部分,你可以象上述那样直接调用Ant 任务。
4.0.0 ... ... maven-antrun-plugin generate-sources classname="org.hibernate.tool.ant.HibernateToolTask" classpathref="maven.dependency.classpath"/> run
在Hibernate 3中使用hbm2java Hbm2java 工具已经经受了Hibernate 3 的考验。hbm2java任务与其它相似的任务一起,被集成进了新版Hibernate 工具集中的hibernatetool任务(撰写此文时仍是alpha版)。Ant任务需要在类路径(class path)中查找以下的.jar 文件: ? hibernate-tools.jar ? velocity-1.4.jar ? velocity-tools-generic-1.4.jar ? jtidy-r8-21122004.jar ? hibernate3.jar ? JDBC drivers 这样一来,Ant任务就要作如下申明:
classname="org.hibernate.tool.ant.HibernateToolTask" classpathref="maven.dependency.classpath"/>最后,你在hibernatetool 任务中调用hbm2java任务,做法如下:
classname="org.hibernate.tool.ant.HibernateToolTask" classpathref="maven.dependency.classpath"/>注意:尽管Hibernate 3版的工具是很有前途的,但它仍只是在alpha阶段,因此使用时务必小心为是。(译注:翻译此文时最新的Hibernate版本是3.1.2,Hibernate Tools的版本是3.1 beta 4) 定制生成的Domain类 现在你知道了怎样从Hibernate映射生成Java源代码, 那么下一步呢?为了更加具体地讨论,我们将使用如图2 和3所描述的简单类模型。这个类模型表示Employee数据库。每个employee赋予一country,讲一种或多种language。每个 country 也有一组国际机场(airport)。 图2. 示例应用程序的UML类图 图3. 示例中所用的数据库模式 有时,你可能想添加域逻辑到Domain类中。实际上,对多数人来说,生成Java类的主要缺点是Domain类变得相对被动,而且把业务逻辑方法加入 生成的Domain类中并不容易,就如所论证的那样使它们更不接近“面向对象“。对这问题并没有一个万能的解决方案,但我们在此描述了一些可能的方法。 把代码放入映射文件: class-code元属性(meta attribute) 对于简单的方法,你可以用class-code元属性来在Hibernate映射文件中指定额外的Java代码。例如,假设我们想在country和 airports间建立双向关系。只要我们想增加airport属性到Country 对象,我们就要确定在Airport对象中相应地增加country属性。我们可以这么做:
true /** * Add an airport to this country */ public void addAirport(Airport airport) { airport.setCountry(this); if (airports == null) { airports = new java.util.HashSet(); } airports.add(airport); } ]]-->除了那些非常小的方法,这种做法并不是特别地令人满意:在XML文件里编写Java代码的做法很容易出错而且难于维护。 使用SQL表达式有时业务逻辑(特别地如果引入了聚合,求和,合计等运算)可能会更自然地用SQL表达式来定义:
formula="(select sum(item.amount) from item where item.order_id = order_id)" />在多数情况下这是非常好的解决方案。但是,必须注意的是每次Hibernate从数据库加载对象时都会执行查询的动作,这可能会对性能造成影响。 使用基类 你也可以用generated-class元属性来定义基类,hbm2java会用这些元属性生成基类,这样你就可以在所生成基类的子类中自由地添加业务逻辑。例如,我们可以在Country类上应用这种技术,就象下面所描述的:
true CountryBase protectedhbm2java生成CountryBase类,此类包含了所有在映射文件中描述的属性,getter,setter等。接着,你可以在Country的派生类中自由地添加业务逻辑,这些会被Hibernate使用并实例化,例如:
public class Country extends CountryBase { /** * Add an airport to this country */ public void addAirport(Airport airport) { airport.setCountry(this); if (getAirports() == null) { setAirports(new java.util.HashSet()); } getAirports().add(airport); } }
包装(Wrapper)或委派(Delegate)模式(Pattern)
对于更复杂的业务逻辑,你或许也想使用下列技术中的一种: ? 你可以定义“wrapper”或“delegate”类,这些类含有作为属性的Domain类,并且提供了指定域类额外的业务逻辑。 ? 你可能更喜欢“服务(service)”或“外观(façade)”方法,“façade”对象(如无状态的会话EJB)提供了一套相关的业务服务来操作Domain对象。 结论 本文描述了我们用来管理Hibernate映射的一种方法, 这在我们所特定的环境中工作得很好。 当然,还有许多别的方法。或许本文会给你的项目提供一些想法,但无论你做什么, 总要使用最适合你项目的方法。 资源 ? Hibernate ? Hibernate Tools ? Ant ? Hibernate Ant tool ? Maven John Ferguson Smart从1991起涉足IT行业, 99年开始从事J2EE的开发。版权声明:任何获得Matrix授权的网站,转载时请务必保留以下作者信息和链接
原文:http://www.javaworld.com/ 译文:http://www.matrix.org.cn/来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/374079/viewspace-131302/,如需转载,请注明出处,否则将追究法律责任。
转载于:http://blog.itpub.net/374079/viewspace-131302/