|
bromon原创 请尊重版权 对象关系映射(Object Relative Mapping)简称ORM,是面向对象开发的一个热点,用来解决JDBC开发中手动进行OR映射的繁杂与不便。EJB中的实体Bean在这个领域是很著名的——既因为它的先进而著名,也因为它的低效而著名。有过实体Bean开发经验的人可能都会为实现远程接口造成的效率低下而头痛,在很多不大不小的项目中,使用实体Bean是否得不偿失,争论很大。一个轻量级的持久化方案也许能够解决一些问题,Hibernate应此而生。 Hibernate是一个中间层,它的目的是把数据库中的关系通过一定的规则映射成为对象,让Java开发人员不用太多的考虑底层数据库的问题,只需要像通常情况下管理对象一样的管理数据。在关系数据库仍将持续占据市场的情况下,它很可观。在数据持久化领域,即便是轻量级的方案也会是复杂饶舌的,也许如同周杰伦的音乐一样不知所云。在学习它之前,最好先回想一下以前进行数据库开发中遇到的问题和不便,想想为什么需要一个持久化层,才能知道很多操作的目的是什么,以及为什么要这么干,在这个问题上我不想做更多的叙述,因为“长久以来……”这样的句式通常long(不好意思,打不出来)长,会对我的键盘和热情造成很大的磨损。如果让我写一本书,那么我会乐意去叙述什么是数据持久化,它有什么好处等等。废话少说,来了。 首先需要配置环境,下载Hibernate 2.1(www.hibernate.org),把lib下的*.jar添加到classpath,你的数据库JDBC驱动程序也应该在classpath中。打开hibernate.properties,针对你使用的数据库,配置相应的信息,比如我使用的是MS SQL Server,配置如下: ## MS SQL Server hibernate.dialect net.sf.hibernate.dialect.SQLServerDialect hibernate.connection.driver_class com.microsoft.jdbc.sqlserver.SQLServerDriver hibernate.connection.url jdbc:microsoft:sqlserver://localhost:1433;DatabaseName=zizz hibernate.connection.username sa hibernate.connection.password 其中很大部分是已经写好的,只需要取掉注释即可,我自己只是修改了数据库名称、帐号、密码。建立一个名为zizz的数据库备用。 然后把这个文件拷贝到你的应用的根目录下。 我们谈论了很多次映射,应该首先来看看这个映射是如何完成的。假设一个最简单的应用,写一个功能最单一的留言板,设计的数据有留言的编号、留言者名称、留言内容,还有留言时间。足够简单吧,换做是你打算怎么干?我猜你要首先建立一个数据库表格,名字也许叫做guestbook。No,这不是面向对象的方式,不妨首先从对象的角度来考虑。我们当然希望每一条留言都以对象的方式呈现,每个对象应该具有的属性有:id、author、content、time。偷个懒,没有画UML。下面这个类应该是很容易理解的: //GuestBook.java package org.bromon.zizz; import java.util.*; public class GuestBook { private int id; private String author; private String content; private Calendar time; private void setId(int id) { this.id=id; } public int getId() { return(id); } public void setAuthor(String author) { this.author=author; } public String getAuthro() { return(author); } public void setContent(String content) { this.content=content; } public String getContent() { return(content); } public void setTime(Calendar time) { this.time=time; } public Calendar getTime() { return(time); } } 基本上是最简单的Bean了,如果觉得困难的话,请你先回火星等我。 需要注意的是setId方法被指定为private,这是因为我希望用这个字段做主键,它最好由系统自动生成,所以不应该由用户来指定,这个方法专为Hibernate准备,所以是私有的。 如何把这个类与数据库映射起来?看看Hibernate的魔法,使用一个XML文件来描述,它应该被命名为GuestBook.hbm.xml: < ?xml version="1.0"? > < !DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 2.0//EN" "' target=_blank >http://hibernate.sourceforge.net/hibernate-mapping-2.0.dtd" > < hibernate-mapping package="org.bromon.zizz" > < class name="GuestBook" table=”guestbook" lazy="true" > < id name="id" type="integer" unsaved-value="null" > < column name="id" sql-type="int" not-null="true"/ > < generator class="identity"/ > < /id > < property name="author" column="author" not-null="true" unique="false"/ > < property name="content" column="content" not-null="true"/ > < property name="time" column="time" not-null="true"/ > < /class > < /hibernate-mapping > 虽然有点陌生,但是很易读,仔细琢磨一下。 下面来编写我们的应用,它的功能是插入数据: //Operate.java package org.bromon.zizz; import net.sf.hibernate.*; import net.sf.hibernate.cfg.*; import net.sf.hibernate.tool.hbm2ddl.*; import java.util.*; public class Operate { public static void main(String args[]) { try { Configuration cfg=new Configuration().addClass(GuestBook.class); SessionFactory sessions=cfg.buildSessionFactory(); new SchemaExport(cfg).create(true,true); Session session=sessions.openSession(); GuestBook gb=new GuestBook(); gb.setAuthor(“Bromon”); gb.setContent(“留言的内容”); gb.setTime(Calendar.getInstance()); Transaction ts=session.beginTransaction(); session.save(gb); ts.commit(); session.close(); }catch(Exception e) { System.out.println(e); } } } 编译吧:javac –d . *.java 执行一下:java org.bromon.zizz.Operate 到数据库里面看看,表格已经建立好了,并且数据也已经保存。如果把 new SchemaExport().create(true,true); 注释掉,那么系统不会创建表格,而只是在已有的表格中添加新的记录,当然,如果表格不存在的话,会产生异常。 你已经看到了Hibernate神奇魔法的5%,它足够的复杂强大,可以让你应付复杂的应用。
one-to-one关系 在绝大多数系统当中不可能只存在一个数据表格,否则就违背了关系数据库的初衷。表与表的关系比较复杂,可以分为几种情况: ● 一对一关联(one to one) ● 一对多关联(one to many) ● 多对一关联(many to one) ● 多对多关联(many to many) 按顺序来讲。假设一个一对一关联的例子是: 表格:person id 编号(主键) name 姓名 email email地址 表格:spouse id 编号(外键) name 姓名 person这个表保存用户信息,而spouse保存用户配偶的信息。在一般情况下一个人只有一个配偶,这很适合我们一对一的情况。如果你对婚外恋感兴趣的话,我们可以在一对多和多对一的关联中讨论这个问题,也许还可以在多对多中^_^(禽兽!)。 OK,下面设计POJO: Person这个类非常简单: /* * Created on 2004-4-19 */ package org.bromon.zizz; /** * @author Bromon */ public class Person { private int id; private String name; private String email; public void setId(int id) { this.id=id; } public int getId() { return(id); } public void setName(String name) { this.name=name; } public String getName() { return(name); } public void setEmail(String email) { this.email=email; } public String getEmail() { return(email); } } 然后编写它的映射规则,这个应该能够理解了: < ?xml version="1.0"? > < !DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 2.0//EN" "' target=_blank >http://hibernate.sourceforge.net/hibernate-mapping-2.0.dtd" > < hibernate-mapping package="org.bromon.zizz" > < class name="Person" table="person" lazy="true" > < id name="id" type="integer" unsaved-value="null" > < column name="id" sql-type="int" not-null="true"/ > < generator class="identity"/ > < /id > < property name="name" column="name" not-null="true" unique="false"/ > < property name="email" column="email" not-null="false"/ > < /class > < /hibernate-mapping > so easy是不是?一切都按部就班。下面是Souse类: /* * Created on 2004-4-20 */ package org.bromon.zizz; /** * @author Bromon */ public class Spouse { private int id; private String name; private Person person; public void setId(int id) { this.id=id; } public int getId() { return(id); } public void setName(String name) { this.name=name; } public String getName() { return(name); } public void setPerson(Person person) { this.person=person; } public Person getPerson() { return(person); } } 注意里面的域person。它的映射文件: < ?xml version="1.0"? > < !DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 2.0//EN" "' target=_blank >http://hibernate.sourceforge.net/hibernate-mapping-2.0.dtd" > < hibernate-mapping package="org.bromon.zizz" > < class name="Spouse" table="spouse" lazy="true" > < id name="id" type="integer" unsaved-value="null" > < column name="id" sql-type="int" not-null="true"/ > < generator class="foreign" > [1] [2] 下一页
|