兩個(gè)對(duì)象之間是一對(duì)一的關(guān)系,如
Person<-->IdCard
有兩種策略可以實(shí)現(xiàn)一對(duì)一的關(guān)聯(lián)映射
主鍵關(guān)聯(lián):即讓兩個(gè)對(duì)象具有相同的主鍵值,以表明它們之間一一對(duì)應(yīng)的關(guān)系;數(shù)據(jù)庫(kù)表不會(huì)有額外的字段來維護(hù)它們之間的關(guān)系累盗,僅通過表的主鍵來關(guān)聯(lián)。(工程hibernate_one2one_pk_1
和hibernate_one2one_pk_2
)
唯一外鍵關(guān)聯(lián):外鍵關(guān)聯(lián)突琳,本來是用于多對(duì)一的配置若债,但是如果加上唯一的限制之后,也可以用來表示一對(duì)一的關(guān)聯(lián)關(guān)系拆融。(工程hibernate_one2one_ufk_1
和hibernate_one2one_ufk_2
)第一種策略中蠢琳,person的主鍵來源于IdCard的主鍵。同時(shí)不管是單向還是雙向镜豹,都是采用的主鍵關(guān)聯(lián)傲须。
第二種策略中是采用外鍵關(guān)聯(lián)的。所謂外鍵關(guān)聯(lián)就是需要額外的添加一個(gè)字段趟脂。同樣的使用外鍵關(guān)聯(lián)我們也分成單向和雙向的泰讽。
一、單向主鍵關(guān)聯(lián)(工程hibernate_one2one_pk_1
)
相關(guān)映射:
首先在數(shù)據(jù)庫(kù)中新建一個(gè)數(shù)據(jù)庫(kù)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肄渗,因?yàn)槲覀冞@個(gè)主鍵是參照IdCard的主鍵 -->
<param name="property">idCard</param><!-- 這個(gè)idCard就是Person類中的一個(gè)屬性 -->
</generator>
</id>
<property name="name"/>
<!-- constrained表示一個(gè)外鍵約束 -->
<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>
測(cè)試:
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);
//注意:這里不會(huì)出現(xiàn)TransientObjectException異常,因?yàn)橐粚?duì)一主鍵關(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);
}
}
}
測(cè)試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 實(shí)體類映射_572131.png . . .]
HibernateUtils.closeSession(session);
}
}
說明:可以看到我們可以通過Person查詢出其對(duì)應(yīng)的身份證IdCard的相關(guān)信息咬最,但是也可以發(fā)現(xiàn)翎嫡,這里我們是不能通過IdCard查詢到Person的,所以需要用到雙向關(guān)聯(lián)永乌。
二惑申、雙向主鍵關(guān)聯(lián)映射(工程hibernate_one2one_pk_2
)
相關(guān)映射:
我們可以看到只是對(duì)象IdCard中維護(hù)了一個(gè)Person對(duì)象,這里我們還是新建一個(gè)數(shù)據(jù)庫(kù)hibernate_one2one_pk_2
翅雏,再自動(dòng)生成相應(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,因?yàn)槲覀冞@個(gè)主鍵是參照IdCard的主鍵 -->
<param name="property">idCard</param><!-- 這個(gè)idCard就是Person類中的一個(gè)屬性 -->
</generator>
</id>
<property name="name"/>
<!-- constrained表示一個(gè)外鍵約束 -->
<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>
測(cè)試:
這里一些測(cè)試都和上一個(gè)例子類似望几,只是這里我們可以通過身份證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í)體類之間的映射沒變橄抹,但是數(shù)據(jù)庫(kù)表之間的映射變?yōu)橥怄I約束了(idCard作為一個(gè)外鍵)靴迫。還是新建一個(gè)數(shù)據(jù)庫(kù)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"/>
<!-- 這里本來是多對(duì)一,但是加上unique約束之后就變成一對(duì)一了 -->
<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>
測(cè)試:
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);
//注意:這里又不能成功保存了楼誓,因?yàn)镮dCard是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);
}
}
}
說明:這里我們又可以看到如果直接保存用戶是不能成功保存的玉锌,這里我們同樣可以使用第二種方式解決,同時(shí)也可以配置cascade="all"
解決疟羹。而在測(cè)試load方法中也只能單向查詢主守,不能雙向查詢,這里我們需要使用雙向外鍵關(guān)聯(lián)榄融。
四参淫、雙向外鍵關(guān)聯(lián)(工程hibernate_one2one_ufk_2
)
相關(guān)映射:
新建數(shù)據(jù)庫(kù)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"/>
<!-- 這里本來是多對(duì)一剃袍,但是加上unique約束之后就變成一對(duì)一了 -->
<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>
測(cè)試:這里其他一些測(cè)試和上面例子一樣,只是這里我們可以進(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ù)庫(kù)中是不會(huì)生成多余的字段的畏邢。這需要將其主鍵指定關(guān)聯(lián)的外鍵是idCard即可业扒。