Hibernate中的关联(Association)映射主要有三种:一对一关联,一对多(或多对一)关联,多对多关联。每种关联都可以分为单向和双向两种。
这篇文章主要说的是在Hibernate中的一对一关联,可以参考Hibernate官方文档的第5章
。至于环境设置,可以参考这个系列的前面几篇文章。
一对一关联有两种映射方式:一种是使用主键关联,限制两个数据表的主键使用相同的值;另一种是一个外键和一个惟一关键字对应。
这里用关联映射将上篇文章
《组件映射》改写了一下,一个人对应一个地址,这是一个单向关联,先谈谈如何使用主键关联。
1. 创建项目
· 新建一个Java项目:AssociationMapping,注意选中“创建单独的源文件夹和输出文件夹”,同时添加“用户库”:hibernate。
2. 编写类文件
· 新建一个类,包名:javamxj.hibernate.association.one2one,类名:Person。
Person.java
/* * Hibernate - 关联(Association)映射 * 创建日期 2005-4-25 * @author javamxj(分享java快乐) * @link Blog: htpp://blog.csdn.net/javamxj/ * htpp://javamxj.mblogger.cn */ package javamxj.hibernate.association.one2one; /** * @hibernate.class table = "T_Person" */ public class Person { private Long id; private String username; private Address address; /** * @hibernate.id * generator-class="foreign" * @hibernate.generator-param * name="property" * value="address" */ public Long getId() {return id;} public void setId(Long id) {this.id = id;} /** * @hibernate.property * length="15" * not-null="true" */ public String getUsername() {return username;} public void setUsername(String username) {this.username = username;} /** * @hibernate.one-to-one * cascade="all" */ public Address getAddress() {return address;} public void setAddress(Address address) {this.address = address;} }
· 注意Person类的主键生成方式是“foreign”,使其参考Address类的主键。
· 在最后几行中指明了如何引用Address类的方式,其中
cascade="all" 表示所有情况下均进行级联操作。
· 这里将Address类独立映射为一个表。
Address.java
package javamxj.hibernate.association.one2one; /** * @hibernate.class table = "T_Address" */ public class Address { private String country; private String city; private String street; private String zipCode; private Long id; public Address() {} public Address(String country, String city, String street, String zipcode) { super(); this.country = country; this.city = city; this.street = street; this.zipCode = zipcode; } /** * @hibernate.id * generator-class="hilo" * unsaved-value="null" */ public Long getId() { return id; } public void setId(Long id) { this.id = id; } /** * @hibernate.property * length = "12" */ public String getCity() {return city;} public void setCity(String city) {this.city = city;} /** * @hibernate.property * length = "12" */ public String getCountry() {return country;} public void setCountry(String country) {this.country = country;} /** * @hibernate.property * length = "6" */ public String getZipCode() {return zipCode;} public void setZipCode(String number) {this.zipCode = number;} /** * @hibernate.property * length = "12" */ public String getStreet() {return street;} public void setStreet(String street) {this.street = street;} }
3. 运行任务
· 复制《
Eclipse快速上手Hibernate--4. 继承映射(1)》文中的build.xml到项目根目录下。
· 双击“generate-hbm”任务,会发现在包中产生了Person.hbm.xml和Address.hbm.xml文件,在src目录下会多了一个hibernate.cfg.xml文件,如果没有,按F5键刷新一下。
· 看看Person类的映射文件中的重点部分:
<generator class="foreign">
<param name="property">address</param>
<!--
To add non XDoclet generator parameters, create a file named
hibernate-generator-params-Person.xml
containing the additional parameters and place it in your merge dir.
-->
</generator>
<param name="property">address</param>
<!--
To add non XDoclet generator parameters, create a file named
hibernate-generator-params-Person.xml
containing the additional parameters and place it in your merge dir.
-->
</generator>
<one-to-one
name="address"
class="javamxj.hibernate.association.one2one.Address"
cascade="all"
outer-join="auto"
constrained="false"
/>
name="address"
class="javamxj.hibernate.association.one2one.Address"
cascade="all"
outer-join="auto"
constrained="false"
/>
· 运行MySql服务器,然后双击“schemaexport”任务,在项目根目录下,会产生一个“schema-export.sql”文件。
· 切换到数据库中,会发现已经自动产生了数据表T_Person和T_Address。
4. 测试程序
· 好了,在包javamxj.hibernate.association.one2one下新建一个Demo.java文件。
Demo.java
package javamxj.hibernate.association.one2one; import net.sf.hibernate.HibernateException; import net.sf.hibernate.Session; import net.sf.hibernate.SessionFactory; import net.sf.hibernate.Transaction; import net.sf.hibernate.cfg.Configuration; public class Demo { public static void main(String[] args) throws HibernateException { Address address1 = new Address("中国", "上海", "普陀", "200055"); Person p1 = new Person(); p1.setUsername("JavaMXJ"); p1.setAddress(address1); Address address2 = new Address("中国", "北京", "海淀", "100086"); Person p2 = new Person(); p2.setUsername("张三"); p2.setAddress(address2); SessionFactory sf = new Configuration().configure().buildSessionFactory(); Session sess = sf.openSession(); Transaction tx= sess.beginTransaction(); sess.save(p1); sess.save(p2); tx.commit(); sess.close(); sf.close(); } }
· 运行这个类,数据表中生成如下数据:
注意这两个表中的主键值相同的。
· 最后的项目结构如下:
5. 另一种映射方式
这里简单说说一个外键和一个惟一关键字对应的方式。
以上面的文章为基础,只要改动一下Person.java即可。
Person.java
/* * Hibernate - 关联(Association)映射 * 创建日期 2005-4-25 * @author javamxj(分享java快乐) * @link Blog: htpp://blog.csdn.net/javamxj/ * htpp://javamxj.mblogger.cn */ package javamxj.hibernate.association.one2one; /** * @hibernate.class table = "T_Person" */ public class Person { private Long id; private String username; private Address address; /** * @hibernate.id * generator-class="hilo" * unsaved-value="null" */ public Long getId() {return id;} public void setId(Long id) {this.id = id;} /** * @hibernate.property * length="15" * not-null="true" */ public String getUsername() {return username;} public void setUsername(String username) {this.username = username;} /** * @hibernate.many-to-one * column = "fk_Address" * cascade="all" * unique="true" */ public Address getAddress() {return address;} public void setAddress(Address address) {this.address = address;} }
· 需要改动的部分用斜体字标出来了,有两处,一处是定义主键的生成方式,一处是Person类如何引用Address类。
· 看看Person类的映射文件中的相关部分:
<generator class="hilo">
<!--
To add non XDoclet generator parameters, create a file named
hibernate-generator-params-Person.xml
containing the additional parameters and place it in your merge dir.
-->
</generator>
<!--
To add non XDoclet generator parameters, create a file named
hibernate-generator-params-Person.xml
containing the additional parameters and place it in your merge dir.
-->
</generator>
<many-to-one
name="address"
class="javamxj.hibernate.association.one2one.Address"
cascade="all"
outer-join="auto"
update="true"
insert="true"
access="property"
column="fk_Address"
unique="true"
/>
cascade="all"
outer-join="auto"
update="true"
insert="true"
access="property"
column="fk_Address"
unique="true"
/>
· 注意:这里<many-to-one>上加上unique="true",表示限制一个Person有一个独有的Address。
· 然后同上面文章一样,生成映射文件,生成表,执行Demo,最后数据表中生成如下数据:
· t_person表中多了一个fk_Address外键,这个外键将t_person表和t_address表一一对应起来了。