【Hibernate步步为营】--双向关联一对一映射具体解释(一)

一对一的映射在对象模型中是常常见到的,为了将对象模型转换为关系模型就必须在映射文件里进行配置,上篇文章讨论了一对一映射的单向关联的情况,重点是<one-to-one>标签的使用,须要在映射的主对象中加入该标签,并将该对象的主键设置为foreign这样就实现了主键关联映射。讨论完了单向接下来讨论双向映射。

一、双向主键关联

双向的主键关联事实上是单向一对一主键关联的一种特殊情况。仅仅只是要在关联对象的两端的映射文件里都要进行<one-to-one>的配置。另外还要在主映射的主键一端採用foreign外键关联属性。

这里相同使用Person和IdCard来讨论,一个人相应着一个唯一的身份证,并且一个身份证也唯一映射着一个人。所以这就产生了双向的关联关系。Person的主键相同也是IdCard的主键,各自是主键的同一时候也是外键,这样的关联关系成为双向一对一映射。表现到关系模型中可例如以下图:

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvemhhbmdfeGlueGl1/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" >

图中的两个表採用了主键关联,person的主键是idCard的主键。所以它们之间构成了朱外键的约束关系,而且保证唯一性,映射到对象模型中。转变为person类和idCard类的一对一关系。例如以下图:

这样的一对一的关系上篇文章中也有讲到用的是<one-to-one>标签,另外这样的一对一映射又是双向的,所以要在两个对象之间同一时候配置<one-to-one>,首先来看idCard相应的类代码和映射文件代码。

1、IdCard相应的信息

IdCard.java类,IdCard类和Person类之间有一对一的关联关系所以要在IdCard类中加入相应的Person属性。这是为了能在映射文件里的外键中加入相应的属性,设置相应的外键关联类。

package com.src.hibernate;

public class IdCard {

	//id属性
	private int id;
	public int getId() {
		return id;
	}
	public void setId(int id) {
		this.id = id;
	}

	//卡号属性
	private String cardNo;
	public String getCardNo() {
		return cardNo;
	}
	public void setCardNo(String cardNo) {
		this.cardNo = cardNo;
	}

	//卡号相应的人
	private Person person;
	public Person getPerson(){
		return person;
	}
	public void setPerson(Person person){
		this.person=person;
	}
}

IdCard.hbm.xml映射文件。在映射文件里加入外键属性person。并加入相应的<one-to-one>标签。目的是强制约束person类来实现一对一的映射关系,最后在映射中将constrained属性设为true,保证强制约束关系。

<?

xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<!-- Generated 2014-5-15 23:47:00 by Hibernate Tools 3.4.0.CR1 -->
<hibernate-mapping>
    <class name="com.src.hibernate.IdCard" table="IDCARD">
        <id name="id" type="int" column="personId">
        	<generator class="foreign">
        		<param name="property">person</param>
        	</generator>
        </id>

        <property name="cardNo" type="string" column="cardno"></property>
        <one-to-one name="person" constrained="true"></one-to-one>
    </class>
</hibernate-mapping>

2、Person相应的信息

Person.java类。在该类中除了加入主要的属性外还要加入相应的IdCard类作为属性,由于它们之间是一对一的双向关联关系,所以在Person类中相同要加入IdCard类,相同的道理IdCard类中相同加入了Person类属性。

package com.src.hibernate;

public class Person {

	//id号
	private int id;
	public int getId() {
		return id;
	}
	public void setId(int id) {
		this.id = id;
	}

	//姓名
	private String name;
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}

	//idCard
	private IdCard idcard;
	public IdCard getIdcard() {
		return idcard;
	}
	public void setIdcard(IdCard idcard) {
		this.idcard = idcard;
	}
}

Person.hbm.xml映射文件,该文件里主键生成策略没有特殊的要求。由于它和IdCard类相互制约的关系,它的主键和外键都是IdCard的主键,另外由于是一对一关系所以要在映射文件里加入<one-to-one>标签来标示。

<?

xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<!-- Generated 2014-5-15 23:47:00 by Hibernate Tools 3.4.0.CR1 -->
<hibernate-mapping>
    <class name="com.src.hibernate.Person" table="PERSON">
        <id name="id" type="int" column="personId">
            <generator class="native"></generator>
        </id>

        <property name="name" type="string" column="personName"></property>
	<!--
	one-to-one标签指示Hibernate怎样载入其关联对象。默认依据主键载入,也就是拿到关系字段值,依据对端的主键来载入关联对象
	 -->
	<one-to-one name="idcard"></one-to-one>
    </class>
</hibernate-mapping>

3、Hibernate映射文件

上面的类和映射文件配置好后接下来要在Hibernate.cfg.xml中配置与数据库映射的信息。须要将两个配置文件加入到Hibernate配置文件里。这样在生成相应的数据库时才干找到相应的生成项。

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-configuration PUBLIC
		"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
		"http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
    <session-factory>
        <property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property>
        <property name="hibernate.connection.url">jdbc:mysql://localhost:3306/hibernate_one2one_pk1</property>
        <property name="hibernate.connection.username">root</property>
        <property name="hibernate.connection.password">1234</property>
        <property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property>

        <mapping resource="com/src/hibernate/Person.hbm.xml"/>
        <mapping resource="com/src/hibernate/IdCard.hbm.xml" ></mapping>
    </session-factory>
</hibernate-configuration>

4、生成结果

配置完毕后就能够将上面的内容生成对应的数据库了,在数据库中它会依照配置的内容生成对应的表结构,在表中有对应的外键和主键字段。生成表结构时Hibernate会在控制台输出对应的SQL语句,例如以下:

alter table IDCARD drop foreign key FK806F76ABAC038CD8
drop table if exists IDCARD
drop table if exists PERSON
create table IDCARD (personId integer not null, cardno varchar(255), primary key (personId))
create table PERSON (personId integer not null auto_increment, personName varchar(255), primary key (personId))
alter table IDCARD add index FK806F76ABAC038CD8 (personId), add constraint FK806F76ABAC038CD8 foreign key (personId) references PERSON (personId)

生成的表结构例如以下图:

在两张表中同一时候生成了personId主键,而且也是对应的外键,它同一时候限制约束了两张表的主键同样且唯一。

5、写入载入測试

生成表后測试下对表的写入和从表中读取数据,编写相应的測试类,測试採用的是单元測试。编写相应的測试方法。

5.1 写入測试

在写入到数据库时一定要注意写入的两个对象都要转化到相应的Trainent状态,否则会出现状态转化的错误,測试代码例如以下:

public void testSave1(){
	Session session=null;
	try{
		//创建一个会话对象
		session=HibernateUtils.getSession();
		//开启会话事务
		session.beginTransaction();

		//创建person对象,并保存
		Person person=new Person();
		person.setName("zhangsan");
		session.save(person);

		//创建idCard对象,并保存
		IdCard idcard=new IdCard();
		idcard.setCardNo("1111111111111");
		idcard.setPerson(person);
		session.save(idcard);

		//提交事务,改动数据库
		session.getTransaction().commit();

	}catch(Exception e){
		//打印错误信息
		e.printStackTrace();
		//业务回滚
		session.getTransaction().rollback();
	}finally{
		//关闭会话
		HibernateUtils.closeSession(session);
	}
}

插入的数据例如以下图:

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvemhhbmdfeGlueGl1/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" >

5.2 载入測试

编写载入方法,由于关联关系是双向的。所以相应的载入操作应该是通过一端载入还有一端,也就是获取相应的Person类,并通过Person类来获取相应的IdCard信息。相反的也要成立,代码例如以下:

public void testLoad1(){
	Session session=null;
	try{
		//创建一个会话对象
		session=HibernateUtils.getSession();
		//开启会话事务
		session.beginTransaction();

		//获取person对象,并保存
		Person person=(Person)session.load(Person.class,5);
		System.out.println("IdCard.Id: "+person.getIdcard().getId());
		System.out.println("IdCard.cardno: "+person.getIdcard().getCardNo());

		//创建idCard对象,并保存
		IdCard idcard=(IdCard)session.load(IdCard.class, 5);
		System.out.println("Person.Id: "+idcard.getPerson().getId());
		System.out.println("Person.name: "+idcard.getPerson().getName());

		//提交事务。改动数据库
		session.getTransaction().commit();

	}catch(Exception e){
		//打印错误信息
		e.printStackTrace();
		//业务回滚
		session.getTransaction().rollback();
	}finally{
		//关闭会话
		HibernateUtils.closeSession(session);
	}
}

执行以上測试方法,在控制台打印的相关内容例如以下:

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvemhhbmdfeGlueGl1/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" >

结语

文章对双向关联关系做了具体的讨论。重点在于分析这样的双向一对一关系的建立及应用。双向的一对一关系分为主键和外键两种,本文主要讨论了主键的约束关系,这样的约束关系同一时候採用了<one-to-one>和<many-to-one>,下文将会具体讨论外键关联。

时间: 2024-10-25 22:04:02

【Hibernate步步为营】--双向关联一对一映射具体解释(一)的相关文章

【Hibernate步步为营】--双向关联一对一映射详解(二)

很不好意思,有两天时间没有更新博客文章了,不写文章的日子还真是感觉很空洞啊,养成了写文章的恶习想改也改不掉啊.说点题外话,前两天收到一位朋友的私信,邀请笔者写一篇有关OWS的文章,用来研究图标工具的一种技术,很荣幸收到这位朋友的邀请,但是因为这几天开发的项目着急上线所以暂时没有时间去研究,只能等这周末了,利用周末的时间来研究然后更新类似的技术文章. 回到文章的正题,上篇文章讨论了双向主键关联,它其实是一对一主键关联的一种特殊情况,想要实现双向的关联就必须在映射文件的两端同时配置<one-to-o

【Hibernate步步为营】--双向关联一对一映射详解(一)

摘要:本文以我自己设计植物大战僵尸碰撞检测算法为例进行分析,浅谈游戏开发中碰撞检测算法.本文所提及的碰撞算法与数据结构也可应用于跑酷等游戏中. 关键优化技术1:只检测逻辑上可能存在碰撞关系的物体.比如植物大战僵尸中,植物发射的子弹只能打击其所在行的僵尸,因此子弹只需遍历检测本行的僵尸,而不是存在场景中的全部僵尸.对于地铁跑酷,奔牛跑酷,吃金币的碰撞检测算法亦是如此. 关键优化技术2:子弹检测碰撞僵尸,或是僵尸检测碰撞子弹,谁检测谁碰撞谁的问题,这个可能跟具体项目息息相关.可以从算法效率,以及设计

【Hibernate步步为营】--多对多映射具体解释

上篇文章具体讨论了一对多映射,在一对多映射中单向的关联映射会有非常多问题,所以不建议使用假设非要採用一对多的映射的话能够考虑使用双向关联来优化之间的关系,一对多的映射事实上质上是在一的一端使用<many-to-one>标签来标明它们之间的关系,另外还须要在一的一端的对象中使用set标明集合映射. 一.单向多对多 仍然依照前几篇的文章格式来讨论.首先来看对象之间的关系,单向的多对多关系是两个对象之间发生的,比方在人和职位之间,一个人能够有多个职位,并且一个职位也能够由多人来负责,所以它们之间就形

【Hibernate步步为营】--单向关联一对一映射(一)

上篇文章对多对一的关联映射做了详细的分析,它在实现上可以有两种方式,并且这两种方式实现也很简单,关键是标签<many-to-one>的使用,它分别指明了多端和一端的映射关系,这种映射关系既是对象模型中的聚合关系.接下来继续讨论关联映射. 一.唯一外键 唯一外键说的是数据库表中的每一行的外键唯一对应着另一张表中的主键,也就是说一个表的主键作为另一张表的外键,并且它们之间的关系是唯一的,这种反应到关系模型中如下图所示: 上图的两个实体表,分别为人和身份证,很明显的一个人对应着一个身份证.身份证作为

【Hibernate步步为营】--关联映射之多对一

上篇文章讨论了Hibernate的基本映射,一个实体类对应着一张表,在相应的Hibernate Mapping文件中使用<class>标签映射.并且实体类中的普通属性对应着表字段,使用<property>标签映射.另外在构造实体类时应注意:在实体类中应实现无参的默认的构造函数,提供一个标示,建议不要使用final修饰实体类,为实体类生成getter和setter方法,最后介绍了几种主要的主键生成策略,接下来讨论多对一映射. 一.关联映射之多对一 对于多对一关联映射其实很容易理解,在

【Hibernate步步为营】--多对多映射详解

上篇文章详细讨论了一对多映射,在一对多映射中单向的关联映射会有很多问题,所以不建议使用如果非要采用一对多的映射的话可以考虑使用双向关联来优化之间的关系,一对多的映射其实质上是在一的一端使用<many-to-one>标签来标明它们之间的关系,另外还需要在一的一端的对象中使用set标明集合映射. 一.单向多对多 仍然按照前几篇的文章格式来讨论,首先来看对象之间的关系,单向的多对多关系是两个对象之间发生的,比如在人和职位之间,一个人可以有多个职位,而且一个职位也可以由多人来负责,所以它们之间就形成了

hibernate(四) 双向多对多映射关系

序言 莫名长了几颗痘,真TM疼,可能是现在运动太少了,天天对着电脑,决定了,今天下午花两小时去跑步了, 现在继上一章节的一对多的映射关系讲解后,今天来讲讲多对多的映射关系把,明白了一对多,多对多个人感觉还是比较容易的,需要理清楚其数据库关系图,那么你就拿下了它.映射文件的配置还是那么些死东西. --WH 一.小疑问的解答 问题一:到这里,有很多学习者会感到困惑,因为他不知道使用hibernate是不是需要自己去创建表,还是hibernate全自动,如果需要自己创建表,那么主外键这种设置也是自己设

hibernate(六)一对一映射

关系映射是指对象之间的关系,并不是指数据库的关系,关系映射是解决当对象处于以下关系之一时,数据库表该如何映射的问题 (一)一对一单向外键关联 1.注解方式配置 创建一个Husband类和Wife类 Husband类:(getWife方法上加注解OneToOne) package cn.orlion.hibernate.model; import javax.persistence.Entity; import javax.persistence.GeneratedValue; import ja

【SSH进阶之路】Hibernate映射——一对一双向关联映射(六)

上篇博文[SSH进阶之路]Hibernate映射--一对一单向关联映射(五),我们介绍了一对一的单向关联映射,单向是指只能从人(Person)这端加载身份证端(IdCard),但是反过来,不能从身份证端加载人得信息.如图所示: 关键原因在于对象模型具有方向性: 单向:一端只能加载另一端,不能反过来. 双向:两端都可以加载另一端. 问题来了:如何我们想从身份证端(IdCard)加载人(Person),怎么办呢? 下面我们开始介绍一对一的双向关联映射. 映射原理 双向关联映射与单向关联映射的原理是一