單向一對(duì)多關(guān)聯(lián):
主控方映射配置如下:
被動(dòng)方(TAddress)的記錄由Hibernate負(fù)責(zé)讀取,之后存放在主控方(TUser)指定的Collection類型屬性中。對(duì)于one-to-many
關(guān)聯(lián)關(guān)系,我們可以采用java.util.Set類型的Collection,表現(xiàn)在XML映射文件中就是<set>節(jié)點(diǎn)。單向一對(duì)多存在一個(gè)問(wèn)題
,由于是單向關(guān)聯(lián),為了保持關(guān)聯(lián)關(guān)系,我們只能通過(guò)主控方對(duì)被動(dòng)方進(jìn)行級(jí)聯(lián)更新。如果被關(guān)聯(lián)方的關(guān)聯(lián)字段為"NOT
NULL",當(dāng)Hibernate創(chuàng)建或者更新關(guān)聯(lián)關(guān)系時(shí),可能出現(xiàn)約束違例。
由于關(guān)聯(lián)方向是單向的,關(guān)聯(lián)關(guān)系由TUser對(duì)象維持,而被關(guān)聯(lián)的addr對(duì)象本身并不知道自己與哪個(gè)TUser對(duì)象相關(guān)聯(lián),也就
是說(shuō),addr對(duì)象本身并不知道自己的user_id應(yīng)該設(shè)為什么數(shù)值。在使用one-to-many進(jìn)行單向關(guān)聯(lián)時(shí),由于Hibernate實(shí)現(xiàn)機(jī)
制中,采用了2條SQL語(yǔ)句進(jìn)行一次數(shù)據(jù)插入操作,相對(duì)單條insert操作,幾乎是2倍的性能開銷,效率較低,因此,對(duì)于性能
敏感的系統(tǒng)而言,這樣的解決方案所帶來(lái)的開銷可能難以承受。
如果addr對(duì)象知道如何獲取user_id字段的內(nèi)容,那么執(zhí)行insert語(yǔ)句的時(shí)候直接將數(shù)據(jù)植入即可。這樣不但繞開了約束違例
的可能,而且還節(jié)省了一條update語(yǔ)句的開銷,大幅度提高了性能。雙向一對(duì)多關(guān)系的出現(xiàn)解決了這個(gè)問(wèn)題。它除了避免約
束和提高性能的好處之外,還帶來(lái)另外一個(gè)優(yōu)點(diǎn),由于建立了雙向關(guān)聯(lián),我們可以在關(guān)聯(lián)雙方中任意一方,訪問(wèn)關(guān)聯(lián)的另一
方,這提供了更豐富靈活的控制手段。
雙向一對(duì)多關(guān)聯(lián),實(shí)際上是"一對(duì)多"與"多對(duì)一"關(guān)聯(lián)的組合。也就是說(shuō)我們必須在主控方配置單向一對(duì)多關(guān)系的基礎(chǔ)上,在
被控方配置與其對(duì)應(yīng)的多對(duì)一關(guān)系。
上代碼:
package com.wyq.demo.common.reference.onetomany;
import java.io.Serializable;
import java.util.Set;
/**
* @author 作者
* @version 創(chuàng)建時(shí)間:2008-11-28 上午10:10:23
* 類說(shuō)明 在one-to-many關(guān)系中,將many一方設(shè)為主控方(inserse=false)將有助于性能的改善。
*/
public class TUser implements Serializable {
private Integer id;
private Integer age;
private String name;
private Set addresses;
public Set getAddresses() {
return addresses;
}
public void setAddresses(Set addresses) {
this.addresses = addresses;
}
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
對(duì)應(yīng)的映射文件:
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<!--
Mapping file autogenerated by MyEclipse Persistence Tools
-->
<hibernate-mapping>
<class name="com.wyq.demo.common.reference.onetomany.TUser" table="t_user" dynamic-insert="true" dynamic-update="true" catalog="sample">
<id name="id" type="java.lang.Integer">
<column name="id" />
<generator class="native" />
</id>
<property name="name" type="java.lang.String">
<column name="name"></column>
</property>
<property name="age" type="java.lang.Integer">
<column name="age"></column>
</property>
<!-- 控制方向反轉(zhuǎn),Tuser不再作為主控方,而是將關(guān)聯(lián)關(guān)系的維護(hù)工作
交給關(guān)聯(lián)對(duì)象Taddress來(lái)完成,這樣Taddress對(duì)象在持久化過(guò)程中
,就可以主動(dòng)獲取其關(guān)聯(lián)的TUser對(duì)象的id,并將其作為自己的user_id,
之后執(zhí)行一次insert操作就可以完成全部工作。
-->
<set name="addresses" table="t_address" inverse="true" cascade="all" order-by="zipcode asc">
<key column="user_id"></key>
<one-to-many class="com.wyq.demo.common.reference.onetomany.TAddress"/>
</set>
</class>
</hibernate-mapping>
關(guān)聯(lián)對(duì)象:
package com.wyq.demo.common.reference.onetomany;
import java.io.Serializable;
/**
* @author 作者
* @version 創(chuàng)建時(shí)間:2008-11-28 上午10:11:01
* 類說(shuō)明 雙向一對(duì)多關(guān)系除了避免約束違例和提高性能的好處之外,還帶來(lái)了另外一個(gè)優(yōu)點(diǎn)
* 由于建立雙向關(guān)聯(lián),我們可以在關(guān)聯(lián)雙方中任意一方,訪問(wèn)關(guān)聯(lián)的另一方。
*/
public class TAddress implements Serializable {
private Integer id;
private String address;
private String zipcode;
private String tel;
private String type;
private Integer userId;
private TUser user;
public String getAddress() {
return address;
}
public void setAddress(String address) {
this.address = address;
}
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getTel() {
return tel;
}
public void setTel(String tel) {
this.tel = tel;
}
public String getType() {
return type;
}
public void setType(String type) {
this.type = type;
}
public TUser getUser() {
return user;
}
public void setUser(TUser user) {
this.user = user;
}
public Integer getUserId() {
return userId;
}
public void setUserId(Integer userId) {
this.userId = userId;
}
public String getZipcode() {
return zipcode;
}
public void setZipcode(String zipcode) {
this.zipcode = zipcode;
}
}
關(guān)聯(lián)對(duì)象的映射文件:
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<!--
Mapping file autogenerated by MyEclipse Persistence Tools
-->
<hibernate-mapping>
<class name="com.wyq.demo.common.reference.onetomany.TAddress" table="t_address" dynamic-insert="false" dynamic-update="false" catalog="sample">
<id name="id" type="java.lang.Integer">
<column name="id" />
<generator class="native" />
</id>
<property name="address" type="java.lang.String" column="address"></property>
<property name="zipcode" type="java.lang.String" column="zipcode"></property>
<property name="tel" type="java.lang.String" column="tel"></property>
<property name="type" type="java.lang.String" column="type"></property>
<many-to-one name="user" class="com.wyq.demo.common.reference.onetomany.TUser" cascade="none" outer-join="auto" column="user_id" not-null="true"></many-to-one>
</class>
</hibernate-mapping>
Inverse指的是關(guān)聯(lián)關(guān)系的控制方向,而cascade指的是層級(jí)之間的連鎖操作。在one-to-many關(guān)系中,將many一方設(shè)為主控方(inverse=false)將有助性能的改善。
posted on 2009-11-04 16:06
王永慶 閱讀(166)
評(píng)論(0) 編輯 收藏 所屬分類:
HIBERNATE