Hibernate的关联关系映射
(注:以下描述适合
hibernate3.1
版本)
以下所有描述以选课系统这个案例来说明。在一个选课系统中,基本的对象有课题(Course)、教师(Teacher)和学生(Student)。
一、一对一关联(有
共享主键关联和
惟一外键关联两种)
1、共享主键关联
比如课题是由教师去任教,没有教师就没有课题,这样课题就可以完全由教师来决定。
Java示例代码如下:
public
class
Course {
private
Long
id
;
private
String
name
;
private
Teacher
teacher
;
//
省略
setter
和
getter
方法
}
public
class
Teacher {
private
Long
id
;
private
String
name
;
private
Course
course
;
//
省略
setter
和
getter
方法
}
配置文件
Teacher.hbm.xml
<!-- cascade="all"
表明对
Teacher
对象
CRUD
的操作会级联到
Course
对象
-->
<
one-to-one
name
=
"course"
class
=
"powerwind.bean.Course"
cascade
=
"all"
/>
Course.hbm.xml
<!-- id
值参照
teacher
属性而生成
-->
<
id
name
=
"id"
type
=
"java.lang.Long"
column
=
"id"
>
<
generator
class
=
"foreign"
>
<
param
name
=
"property"
>
teacher
</
param
>
</
generator
>
</
id
>
<!-- teacher
属性参照
-->
<
one-to-one
name
=
"teacher"
class
=
"powerwind.bean.Teacher"
constrained
=
"true"
/>
2、惟一外键关联
外键关联需要一个定义外键字段,比如在教师表定义一个课题的外键 course_id。
配置文件
Teacher.hbm.xml
<
many-to-one
name
=
"course"
class
=
"powerwind.bean.Course"
column
=
"course_id"
cascade
=
"save-update"
unique
=
"true"
/>
如果同时在
Coure.hbm.xml加上如下配置则为双向关联。
<
one-to-one
name
=
"teacher"
class
=
"powerwind.bean.Teacher"
property-ref
=
"course"
/>
3、第三种
以上两种一对一关联在Hibernate中是比较常用的。但我用另外一种方法实现时,发现也可行。测试了一下,目前没有发现错误。
思想是和
唯一外键关联相似,但两个表中都需要一个外键,在配置
<one-to-one>的地方用了
<many-to-one>,即两边对用了
<many-to-one> 。
Course.hbm.xml
<
many-to-one
name
=
"teacher"
class
=
"powerwind.bean.Teacher"
column
=
"teacher_id"
unique
=
"true"
/>
Teacher.hbm.xml
<
many-to-one
name
=
"course"
class
=
"powerwind.bean.Course"
column
=
"course_id"
unique
=
"true"
/>
细看起来,应该是一对多关联的配置,只是两边都设置“一”,欺骗了hibernate。在hibernate文档中有这样一句:
基于外键关联的单向一对一关联和单向多对一关联几乎是一样的。唯一的不同就是单向一对一关联中的外键字段具有唯一性约束。
一对多(多对一)的关联是现在中用得比较多的。现实中,一个教师是可以任教多个课程的,这就是典型的一对多关系。(由于关联是有方向的,一对多和多对一的单向关联是不同的,而双向的时候就一样)
假设我们只需要从课题知道该课题的任教教师是谁?而不需要从教师那里知道该都任教哪些课程。只在
Course.hbm.xml中这样设置就可以。
<
many-to-one
name
=
"teacher"
class
=
"powerwind.bean.Teacher"
column
=
"teacher_id"
/>
如果只需要从教师那里得知他的所有课程,可把原本多对多的关联变成一对多(需要连接表Teacher_ Course)。
Teacher.hbm.xml
<
set
name
=
"courses"
table
=
"teacher_course"
cascade
=
"save-update"
>
<
key
column
=
"teacher_id"
/>
<
many-to-many
column
=
"course_id"
unique
=
"true"
class="powerwind.bean.Course"/>
</
set
>
unique
=
"true"是它不同于多对多的地方。
双向关联就是在单向的基础上,再配置多一项(向)。
最常用最标准的配置是这种:
Course.hbm.xml
<
many-to-one
name
=
"teacher"
class
=
"powerwind.bean.Teacher"
column
=
"teacher_id"
/>
Teacher.hbm.xml
<
set
name
=
"courses"
inverse
=
"true"
>
<
key
column
=
"course_id"
not-null
=
"true"
/>
<
one-to-many
class
=
"powerwind.bean.Course"
/>
</
set
>
其中
inverse="
true" 的设置是比较常用的,设置后,当Teacher和Course对象状态均发生变化时,仅参照Course对象状态的变化来更新数据厍。
举个单向多对多关联的例子。比如远程教学,教师不知道他的学生有谁谁谁,只知道有很多学生在远程听课;而学生可以知道他的老师有哪些。这样,不就是单向多对多关联吗?实际上,单向的多对多关联是不完整的双向多对多关联。
如果单向,则只要在
Student.hbm.xml配置。
<
set
name
=
"teachers"
inverse
=
"true"
table
=
"student_teacher"
>
<
key
column
=
"student_id"
/>
<
many-to-many
column
=
"teacher_id"
class
=
"powerwind.bean.Teacher"
/>
</
set
>
如果双向,在上面的基础上,还要在
Teacher.hbm.xml配置。
<
set
name
=
"students"
table
=
"student_teacher"
>
<
key
column
=
"teacher_id"
/>
<
many-to-many
column
=
" student_id"
class
=
"powerwind.bean.Student"
/>
</
set
>
有时候,连接表还会有其它信息的。比如学生与课程的关系表(stuent_course),里面可以加入学生的考试成绩。这样的话,我们可以把这个连接表定义成这样:
student_coures( id,student_id, course_id, score );
这样,双向多对多关联就变成两个一对多的关联。这种情况下,多对多的双方并没有直接拥有对方的集合引用。这里把student_coures 表对应的实体叫 Score (一时想不出,叫成绩单算了)。
Student.hbm.xml
<
set
name
=
"scores"
inverse
=
"true"
>
<
key
column
=
"student_id"
/>
<
one-to-many
class
=
"powerwind.bean.Score"
/>
</
set
>
Teacher.hbm.xml
<
set
name
=
"scores"
inverse
=
"true"
>
<
key
column
=
"teacher_id"
/>
<
one-to-many
class
=
"powerwind.bean.Score"
/>
</
set
>
Score.hbm.xml
<
many-to-one
name
=
"teacher"
column
=
"teacher_id"
class
=
"powerwind.bean.Teacher"
/>
<
many-to-one
name
=
"student"
column
=
"student_id"
class
=
"powerwind.bean.Student"
/>
本文PDF格式文件下载:http://pickup.mofile.com/1321301160427079
或者 here