兩個對象之間是一對一的關(guān)系,如
Person<-->IdCard有兩種策略可以實現(xiàn)一對一的關(guān)聯(lián)映射
主鍵關(guān)聯(lián):即讓兩個對象具有相同的主鍵值,以表明它們之間一一對應(yīng)的關(guān)系;數(shù)據(jù)庫表不會有額外的字段來維護(hù)它們之間的關(guān)系,僅通過表的主鍵來關(guān)聯(lián)。(工程hibernate_one2one_pk_1和hibernate_one2one_pk_2)
唯一外鍵關(guān)聯(lián):外鍵關(guān)聯(lián),本來是用于多對一的配置,但是如果加上唯一的限制之后,也可以用來表示一對一的關(guān)聯(lián)關(guān)系。(工程hibernate_one2one_ufk_1和hibernate_one2one_ufk_2)第一種策略中,person的主鍵來源于IdCard的主鍵。同時不管是單向還是雙向,都是采用的主鍵關(guān)聯(lián)。
第二種策略中是采用外鍵關(guān)聯(lián)的。所謂外鍵關(guān)聯(lián)就是需要額外的添加一個字段。同樣的使用外鍵關(guān)聯(lián)我們也分成單向和雙向的。
一、單向主鍵關(guān)聯(lián)(工程hibernate_one2one_pk_1)
相關(guān)映射:

首先在數(shù)據(jù)庫中新建一個數(shù)據(jù)庫hibernate_one2one_pk_1,然后生成相關(guān)的表。
Person.hbm.xml
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="cn.itcast.hibernate.Person" table="_person">
<id name="id">
<generator class="foreign"><!-- 這里不能使用native,因為我們這個主鍵是參照IdCard的主鍵 -->
<param name="property">idCard</param><!-- 這個idCard就是Person類中的一個屬性 -->
</generator>
</id>
<property name="name"/>
<!-- constrained表示一個外鍵約束 -->
<one-to-one name="idCard" constrained="true"></one-to-one>
</class>
</hibernate-mapping>
IdCard.hbm.xml
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="cn.itcast.hibernate.IdCard" table="_idCard">
<id name="id">
<generator class="native"/>
</id>
<property name="cardNo"/>
</class>
</hibernate-mapping>
測試:
One2OneTest.java
package cn.itcast.hibernate;
import org.hibernate.Session;
import org.junit.Test;
public class One2OneTest {
@Test
public void testSave1(){
Session session = null;
try {
session = HibernateUtils.getSession();
session.beginTransaction();
IdCard idCard = new IdCard();
idCard.setCardNo("1111");
Person person = new Person();
person.setName("張三");
person.setIdCard(idCard);
//注意:這里不會出現(xiàn)TransientObjectException異常,因為一對一主鍵關(guān)聯(lián)映射中
//默認(rèn)了cascade屬性
session.save(person);
session.getTransaction().commit();
} catch (Exception e) {
e.printStackTrace();
session.getTransaction().rollback();
}finally{
HibernateUtils.closeSession(session);
}
}
}
測試load方法:
@Test
public void testLoad1(){
Session session = null;
try {
session = HibernateUtils.getSession();
session.beginTransaction();
Person person = (Person) session.load(Person.class, 1);
System.out.println("person.name = " + person.getName());
System.out.println("idCard.cardNo = " + person.getIdCard().getCardNo());
session.getTransaction().commit();
} catch (Exception e) {
e.printStackTrace();
session.getTransaction().rollback();
}finally{
![Uploading 實體類映射_572131.png . . .]
HibernateUtils.closeSession(session);
}
}
說明:可以看到我們可以通過Person查詢出其對應(yīng)的身份證IdCard的相關(guān)信息,但是也可以發(fā)現(xiàn),這里我們是不能通過IdCard查詢到Person的,所以需要用到雙向關(guān)聯(lián)。
二、雙向主鍵關(guān)聯(lián)映射(工程hibernate_one2one_pk_2)
相關(guān)映射:

我們可以看到只是對象IdCard中維護(hù)了一個Person對象,這里我們還是新建一個數(shù)據(jù)庫hibernate_one2one_pk_2,再自動生成相應(yīng)的表。
Person.hbm.xml
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="cn.itcast.hibernate.Person" table="_person">
<id name="id">
<generator class="foreign"><!-- 這里不能使用native,因為我們這個主鍵是參照IdCard的主鍵 -->
<param name="property">idCard</param><!-- 這個idCard就是Person類中的一個屬性 -->
</generator>
</id>
<property name="name"/>
<!-- constrained表示一個外鍵約束 -->
<one-to-one name="idCard" constrained="true"></one-to-one>
</class>
</hibernate-mapping>
IdCard.hbm.xml
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="cn.itcast.hibernate.IdCard" table="_idCard">
<id name="id">
<generator class="native"/>
</id>
<property name="cardNo"/>
<one-to-one name="person"></one-to-one>
</class>
</hibernate-mapping>
測試:
這里一些測試都和上一個例子類似,只是這里我們可以通過身份證IdCard查詢出Person。
@Test
public void testLoad2(){
Session session = null;
try {
session = HibernateUtils.getSession();
session.beginTransaction();
IdCard idCard = (IdCard) session.load(IdCard.class, 1);
System.out.println("idCard.cardNo = " + idCard.getCardNo());
System.out.println("idcard.peson.name = " + idCard.getPerson().getName());
session.getTransaction().commit();
} catch (Exception e) {
e.printStackTrace();
session.getTransaction().rollback();
}finally{
HibernateUtils.closeSession(session);
}
}
三、單向外鍵關(guān)聯(lián)(工程hibernate_one2one_ufk_1)
相關(guān)映射:

說明:可以看到實體類之間的映射沒變,但是數(shù)據(jù)庫表之間的映射變?yōu)橥怄I約束了(idCard作為一個外鍵)。還是新建一個數(shù)據(jù)庫hibernate_one2one_ufk_1
Person.hbm.xml
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="cn.itcast.hibernate.Person" table="_person">
<id name="id">
<generator class="native"></generator>
</id>
<property name="name"/>
<!-- 這里本來是多對一,但是加上unique約束之后就變成一對一了 -->
<many-to-one name="idCard" unique="true"></many-to-one>
</class>
</hibernate-mapping>
IdCard.hbm.xml
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="cn.itcast.hibernate.IdCard" table="_idCard">
<id name="id">
<generator class="native"/>
</id>
<property name="cardNo"/>
</class>
</hibernate-mapping>
測試:
One2OneTest.java
package cn.itcast.hibernate;
import org.hibernate.Session;
import org.junit.Test;
public class One2OneTest {
@Test
public void testSave1(){
Session session = null;
try {
session = HibernateUtils.getSession();
session.beginTransaction();
IdCard idCard = new IdCard();
idCard.setCardNo("1111");
Person person = new Person();
person.setName("張三");
person.setIdCard(idCard);
//注意:這里又不能成功保存了,因為IdCard是Transient狀態(tài)
session.save(person);
session.getTransaction().commit();
} catch (Exception e) {
e.printStackTrace();
session.getTransaction().rollback();
}finally{
HibernateUtils.closeSession(session);
}
}
@Test
public void testSave2(){
Session session = null;
try {
session = HibernateUtils.getSession();
session.beginTransaction();
IdCard idCard = new IdCard();
idCard.setCardNo("1111");
session.save(idCard);
Person person = new Person();
person.setName("張三");
person.setIdCard(idCard);
//這樣才能成功保存
session.save(person);
session.getTransaction().commit();
} catch (Exception e) {
e.printStackTrace();
session.getTransaction().rollback();
}finally{
HibernateUtils.closeSession(session);
}
}
@Test
public void testLoad1(){
Session session = null;
try {
session = HibernateUtils.getSession();
session.beginTransaction();
Person person = (Person) session.load(Person.class, 1);
System.out.println("person.name = " + person.getName());
System.out.println("idCard.cardNo = " + person.getIdCard().getCardNo());
session.getTransaction().commit();
} catch (Exception e) {
e.printStackTrace();
session.getTransaction().rollback();
}finally{
HibernateUtils.closeSession(session);
}
}
}
說明:這里我們又可以看到如果直接保存用戶是不能成功保存的,這里我們同樣可以使用第二種方式解決,同時也可以配置cascade="all"解決。而在測試load方法中也只能單向查詢,不能雙向查詢,這里我們需要使用雙向外鍵關(guān)聯(lián)。
四、雙向外鍵關(guān)聯(lián)(工程hibernate_one2one_ufk_2)
相關(guān)映射:

新建數(shù)據(jù)庫hibernate_one2one_ufk_2生成相應(yīng)的表。
Person.hbm.xml
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="cn.itcast.hibernate.Person" table="_person">
<id name="id">
<generator class="native"></generator>
</id>
<property name="name"/>
<!-- 這里本來是多對一,但是加上unique約束之后就變成一對一了 -->
<many-to-one name="idCard" unique="true"></many-to-one>
</class>
</hibernate-mapping>
IdCard.hbm.xml
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="cn.itcast.hibernate.IdCard" table="_idCard">
<id name="id">
<generator class="native"/>
</id>
<property name="cardNo"/>
<!-- 這里默認(rèn)是使用person.id去加載,這里我們必須指明是使用idCard去加載 -->
<one-to-one name="person" property-ref="idCard"></one-to-one>
</class>
</hibernate-mapping>
測試:這里其他一些測試和上面例子一樣,只是這里我們可以進(jìn)行雙向查詢。
@Test
public void testLoad2(){
Session session = null;
try {
session = HibernateUtils.getSession();
session.beginTransaction();
IdCard idCard = (IdCard) session.load(IdCard.class, 1);
System.out.println("idCard.cardNo = " + idCard.getCardNo());
System.out.println("idcard.peson.name = " + idCard.getPerson().getName());
session.getTransaction().commit();
} catch (Exception e) {
e.printStackTrace();
session.getTransaction().rollback();
}finally{
HibernateUtils.closeSession(session);
}
}
說明:這里我們可以看到使用雙向關(guān)聯(lián)可以進(jìn)行雙向查詢,但是在IdCard這邊是使用的<one-to-one>標(biāo)簽,在數(shù)據(jù)庫中是不會生成多余的字段的。這需要將其主鍵指定關(guān)聯(lián)的外鍵是idCard即可。