作為一種輕量級的關系映射工具,Hibernate支持各種關系映射,例如:多對一、一對多和一對一的數據庫表關系,通過映射文件的靈活配置即可輕松實現。Hibernate的重要部分就是關系映射。本章我們將會就這幾個關系進行說明介紹:
- 單向多對多映射
- 雙向多對多映射
- 單向多對一映射
- 單向一對多映射
- 雙向一對多(多對一)映射
- 基于外鍵的單向一對一映射
- 基于外鍵的雙向一對一映射
- 基于主鍵的單向一對一映射
- 基于主鍵的雙向一對一映射
這里,在介紹的過程中,會著重介紹配置文件的核心代碼變化,而忽略掉Test測試方法。另外,需要注意的是基于外鍵、主鍵以及無任何的關系與變化。
1.1 單向多對多映射
這里我們使用實例來進行說明:
假設有角色和用戶組兩個表,是多對多的關系,即一個角色可以多個用戶組擁有,一個用戶組也可以擁有多個角色。這里我們在進行編寫的時候呢,就需要增加一個角色-組的對應表,用來記錄多對多的關系。
例如一個網站的用戶角色可以有查看、添加、刪除和修改等功能,用戶又可以分為管理員、版主、注冊用戶和匿名用戶等,那么這里的用戶和角色就是多對多的關系,即一個用戶可以擁有多個角色,每個角色也可以賦給多個用戶。
步驟:
1.建立角色表、用戶表以及角色-用戶表。第三個表主要就是將他們在各自關系表中的主鍵屬性進行創建一個表。
2.建立角色實體類。這里我們需要注意的是將用戶表也進行實例化,也就是在創建的時候,需要建立一個集合類型屬性變量,這樣才能展現出多個用戶。
private Set groups = new HashSet();
3.建立用戶組的實體類
這里與上面的是一致的,也需要將角色表進行實例化。
private Set roles = new HashSet();
4.因為角色端是多對多的控制方,其映射文件可以進行編寫:
<!-- 配置多對多關系映射 -->
<set name="groups" cascade="save-update">
<key column="role_id"></key>
<many-to-many class="*" column="group_id"></many-to-many>
</set>
5.Group(用戶組)是被控制方,在進行編寫的時候,我們不需要進行特別的主要,就像原來的那么編寫即可。
6.編寫測試類Test
這里筆者就不一一說明測試類的書寫了,這里我們只需要注意Set集合的編寫方法。
1.2 雙向多對多映射
我們還是使用上面的例子進行說明。
首先,根據上面的步驟,我們需要修改角色端映射文件的代碼書寫。
這里我們需要在Set標簽中再加入一個屬性——inverse。
inverse屬性負責控制關系,默認為false,也就是關系的兩端都能控制,但這樣會造成一些問題。更新時會因為兩端都控制關系,于是重復更新。一般來說有一端要設為true,也就是讓另一端進行控制關系。
第二,我們需要在用戶組映射文件中進行編寫。
<!-- 配置多對多關系映射 -->
<set name="roles" cascade="save-update">
<key column="group_id"></key>
<many-to-many class="*" column="role_id"></many-to-many>
</set>
這樣,我們就比較好的將雙向與單向多對多的映射進行了比較與介紹。
1.3 單向多對一映射
在數據庫表關系中,經常出現多對一的映射,例如一個人只能有一個住址(假定),一個住址可以住多個人。那么人和住址就是多對一的關系。
步驟:
1.建表,人(Person)、住址(address)。
2.建實體類,這里只需要在多的那一端也就是人實體類建立一個住址屬性變量。
private Address address;
而另外一端,則如同正常的實體類一般進行編寫即可。
3.建立Person.hbm.xml映射文件:
<!-- 配置多對一的關聯關系 -->
<many-to-one name="address" column="address_id"></many-to-one>
就能達到控制單向多對一關系映射。
4建立Address.hbm.xml映射文件
不需要進行特別的配置,略過。。。
5.添加測試類Test
1.4單向一對多映射
比較典型的一對多映射就是公司與雇員的關系,即一個公司可以有多個雇員。
單向一對多的映射關系并不是很常見,其配置同單向多對一非常相似,其主動方為“一”的一方,使用one-to-many配置屬性。
步驟:
1.建表。公司表(company)、雇員表(employee)。
2.建公司與雇員的實體類。略。PS:我想有點讀者已經發現了,多與一的時候,我們只需要在一的那邊編寫類的屬性變量。
3.編寫公司映射文件Company.hbm.xml。
<!-- 配置一對多關系映射 -->
<set name="employees" cascade="all">
<key column="com_id"></key>
<one-to-many class="*"></many-to-many>
</set>
4.雇員映射文件Employee.hbm.xml
略過,與一般相同。
5.編寫測試類Test
1.5 雙向一對多(多對一)映射
常見的雙向一對多或者是多對一映射是父子關系
步驟:
1.建立數據庫表。father、son
2.建立實體類
3.映射文件Father.hbm.xml的配置
<!-- 配置一對多關系映射 -->
<set name="sons" cascade="all" incerse="true">
<key column="father_id"></key>
<one-to-many class="*"></many-to-many>
</set>
4.映射文件Son.hbm.xml配置
<!-- 配置多對一的關聯關系 -->
<many-to-one name="father" column="father_id"></many-to-one>
5.編寫測試類Test
1.6 基于外鍵的單向一對一映射
基于外鍵的單向一對一映射,常見的例子是用戶(user)和電子郵件(email),每個用戶只有一個電子郵件,并通過外鍵email_id同電子郵件相對應。但是電子郵件沒有對應的用戶
步驟:
1.建立email、user數據庫表
2.建立實體類。
在User.java中,我們需要定義一個郵件類型變量
private Email email;
3.映射文件User.hbm.xml
<!-- 配置多對一的關聯關系 -->
<many-to-one name="email" column="email_id" class="*" unique="true">
</many-to-one>
在這個映射文件中,我們給many-to-one配置了一個unique屬性。
配置這個屬性的作用是不能有重復的主鍵,就限定了"多"的一方必須變成“一”,這樣就實現了一對一映射。單向的一對一映射,只需要配置一方即可。
4.映射文件Email.hbm.xml
與一般相同即可。
5.編寫測試類Test
1.7基于外鍵的雙向一對一映射
修改前一節單項映射為雙向映射。在保持數據庫表不變的情況下,增加相應的配置信息,實現雙向映射。例如:用戶表與電子郵件就是雙向一對一的映射關系,即每個用戶都有唯一的電子郵件對應,例如常見的電子商務網站淘寶和ebay等,一對一可以通過主鍵對應,也可以通過外鍵來對應。
- one-to-one
- property-ref
步驟:
1.數據庫表結構與上節一樣
2.雙向所以兩個實體類都要進行編寫類屬性變量
3.映射文件User.hbm.xml
<!-- 配置一對一的關聯關系 -->
<many-to-one name="email" column="email_id" unique="true">
</many-to-one>
4.映射文件Email.hbm.xml
<!-- 配置一對一的關聯關系 -->
<one-to-one name="user" property-ref="email">
</one-to-one>
property-ref屬性:指定關聯類的一個屬性,這個屬性將會和本外鍵相對應。如果沒有指定,會使用對方關聯類的主鍵
5.編寫測試類Test
1.8 基于主鍵的單向一對一映射
一對一映射,前面介紹的是基于外鍵的,還有一種基于主鍵的一對一映射,即兩個表的主鍵值相同。單向一對一的情況不多,例如用戶和電子郵件之間的關系,假定沒有用戶必須有唯一的電子郵件與之對應,而電子郵件可以沒有對應的用戶,即對應關系是單向的。
基于主鍵的一對一映射特點是兩個表都以主鍵作為關聯字段,這同基于外鍵的一對一映射有很大的不同,即關聯字段都是各自的主鍵。
步驟:
1.建表。同時都以ID作為主鍵。
2.建立實體類。這里和基于外鍵的不同是單向兩個實體類同時都要建立類屬性變量。
3.Emailpk.hbm.xml映射文件
這里呢?有一個比較重要的屬性。
<id ... >
<column .../>
<generator class="foreign">
<param name-"property">userpk</paprm>
<generator>
</id>
...
<one-to-one name="userpk" class="*" constrained="true">
</one-to-one>
這里,我們引進了幾個其他的屬性。
generator class="foreign" 意味著Emailpk使用外部主鍵,<param name="property">userpk</param>配置了Emailpk使用Usepk的主鍵作為自己的主鍵,這樣就實現了基于主鍵的一對一映射。
constrained="true"說明userpk的主鍵值存在一個約束,即emailpk使用了userpk的主鍵。
4.Userpk.hbm.xml映射文件
與一般相同,不做闡述。
5.編寫測試文件Test
1.9基于主鍵的雙向一對一映射
前面介紹了基于主鍵的單向一對一映射,本節介紹基于主鍵的雙向一對一映射,還是使用前面的數據庫表和相關類。例如用戶和電子郵件之間的關系,每個用戶有唯一的電子郵件,每個電子郵件有唯一的用戶,假如這種對應是通過各自的主鍵實現的,那么就是基于主鍵的雙向一對一映射關系。
步驟:
1.建表。
2.編寫實體類。
3.映射文件Emailpk.hbm.xml
與上面的一樣。
4.映射文件Userpk.hbm.xml
<one-to-one name="emailpk" class="*" constrained="true">
</one-to-one>
5.編寫測試類Test。