Hibernate映射组成关系

  建立域模型和关系数据模型有着不同的出发点。
  域模型(面向对象设计):由程序代码组成,通过细化持久化类的的粒度可提高代码的可重用性,简化编程。
  关系数据模型(数据库设计):在没有数据冗余的情况下,应该尽可能减少表的数目,简化表之间的参照关系,以便提高数据的访问速度。
  
  在这篇文章中我们以Worker(工人)和Pay(薪酬)之间的关系为例进行说明:
  image_1b34cojeh1ua6193h6cg128t9fc9.png-37.2kB
  
  Hibernate 把持久化类的属性分为两种:
  值(value)类型:没有OID,不能被单独持久化,生命周期依赖于所属的持久化类的对象的生命周期。
  实体(entity)类型:有OID,可以被单独持久化,有独立的生命周期。
  
  显然在上述例子中Worker为实体类型,Pay为值类型。如果我们将这两个类根据组成关系映射为数据库中的一张表而不是两张表,那么单表查询的速度是要快于多表查询的。
  Hibernate 使用 <component> 元素来映射组成关系, 该元素表明 pay 属性是 Worker 类一个组成部分, 在 Hibernate 中称之为组件。
  
  下面进行测试,首先新建两个java类:
  

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public class Worker {
private Integer id;
private String name;
private Pay pay;
  //getters and setters
}
public class Pay {
private int monthlyPay;
private int yearPay;
private int vocationWithPay;
//getters and setters
}

生成映射文件Worker.hbm.xml,并在映射文件中映射组成关系:

1
2
3
4
5
6
7
8
<!-- 映射组成关系 -->
<component name="pay" class="Pay">
<parent name="worker"/>
<!-- 指定组成关系的组件的属性 -->
<property name="monthlyPay" column="MONTHLY_PAY"></property>
<property name="yearPay" column="YEAR_PAY"></property>
<property name="vocationWithPay" column="VOCATION_WITH_PAY"></property>
</component>

编写测试程序:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
@Test
public void testComponent(){
Worker worker = new Worker();
Pay pay = new Pay();
pay.setMonthlyPay(1000);
pay.setYearPay(80000);
pay.setVocationWithPay(5);
worker.setName("ABCD");
worker.setPay(pay);
session.save(worker);
}

运行程序,发现生成了一张数据表worker:

image_1b34ddp4h2vf1b6p1p7ji341u32m.png-20.1kB

并且成功插入了记录:

image_1b34dfjo817uu1rg55kf37e1a3o13.png-11.6kB