文档章节

hibernate系列(四)一对一关联关系

乒乓狂魔
 乒乓狂魔
发布于 2015/02/07 10:24
字数 1032
阅读 65
收藏 1
以Person类和IDCard类为例,这里仅仅说一种一对一关联关系,即Person类拥有IDCard,但是IDCard不含Person类,数据库库的表如下:
CREATE TABLE `hibernate`.`person` (
  `id` INT NOT NULL AUTO_INCREMENT,
  `name` VARCHAR(45) NULL,
  `age` INT NULL,
  `idcard_id` INT NULL,
  PRIMARY KEY (`id`));

CREATE TABLE `hibernate`.`idcard` (
  `id` INT NOT NULL AUTO_INCREMENT,
  `number` INT NULL,
  `content` VARCHAR(45) NULL,
  PRIMARY KEY (`id`));

Person类如下:
public class Person {

	private Long id;
	private String name;
	private Long age;
	private IDCard idCard;
//省略get、set方法
}

Person类对应的Person.hbm.xml映射文件为:
<hibernate-mapping>
	<class name="com.ligang.domain.Person" table="person">
		<id name="id" column="id" type="long">
			<generator class="identity"/>
		</id>
		<property name="name" column="name" type="string"/>
		<property name="age" column="age" type="long"/>
		<many-to-one name="idCard" class="com.ligang.domain.IDCard" column="idcard_id" cascade="save-update"></many-to-one>
	</class>
</hibernate-mapping>

虽然是一对一但是,这种形式的一对一就是多对一的特例,所以仍然使用<many-to-one>的标签,其中的name指的是Person类的idCard属性,column指的是person表中的字段名为idcard_id,class指的是将IDCard类对应的表的主键的值作为idcard_id的值。cascade字段表示保存Person类时级联的保存IDCard类。
下面看下IDCard类:

public class IDCard {

	private Long id;
	private Long number;
	private String content;
//省略get、set方法
}

IDCard类对应的IDCard.hbm.xml映射文件为:
<hibernate-mapping>
	<class name="com.ligang.domain.IDCard" table="idcard">
		<id name="id" column="id" type="long">
			<generator class="identity"/>
		</id>
		<property name="number" column="number" type="long"/>
		<property name="content" column="content" type="string"/>
	</class>
</hibernate-mapping>

然后就是测试增添方法:
@Test
	public void addPerson(){
		Session session=hibernateDao.getSession();
		Transaction tx=session.beginTransaction();
		
		Person p=new Person();
		p.setName("张三");
		p.setAge(122L);
		
		IDCard idCard=new IDCard();
		idCard.setNumber(123445L);
		idCard.setContent("你是一个人");
		
		p.setIdCard(idCard);
		
		session.save(p);
		
		tx.commit();
		session.close();
	}

此时就会先保存IDCard对象,然后获取其主键并赋值给person表的idcard_id字段,然后增添Person对象。如下sql:
Hibernate: insert into hibernate.idcard (number, content) values (?, ?)
Hibernate: insert into hibernate.person (name, age, idcard_id) values (?, ?, ?)

更新如下:
@Test
	public void updatePerson(){
		Session session=hibernateDao.getSession();
		Transaction tx=session.beginTransaction();
		
		Person p=(Person) session.get(Person.class,6L);
		p.setName("张三");
		p.setAge(122L);
		
		IDCard idCard=new IDCard();
		idCard.setNumber(123445L);
		idCard.setContent("你是一个人");
		
		p.setIdCard(idCard);
		
		session.save(p);
		
		tx.commit();
		session.close();
	}

此时的sql如下:
Hibernate: select person0_.id as id1_3_0_, person0_.name as name2_3_0_, person0_.age as age3_3_0_, person0_.idcard_id as idcard_i4_3_0_ from hibernate.person person0_ where person0_.id=?
Hibernate: insert into hibernate.idcard (number, content) values (?, ?)
Hibernate: update hibernate.person set name=?, age=?, idcard_id=? where id=?

此时并没有删除原有的IDCard,只是根据Person再也找不到它了。目前我还不知道怎么设置来删除无用的IDCard。

获取:由Person的主键获取IDCard比较容易,但是如果想从IDCard主键获取Person呢?
首先更改IDCard类,添加person属性:

public class IDCard {

	private Long id;
	private Long number;
	private String content;
	private Person person;
//略get、set方法
}

然后更改上述的IDCard.hbm.xml映射文件如下:
<hibernate-mapping>
	<class name="com.ligang.domain.IDCard" table="idcard">
		<id name="id" column="id" type="long">
			<generator class="identity"/>
		</id>
		<property name="number" column="number" type="long"/>
		<property name="content" column="content" type="string"/>
		<one-to-one name="person" property-ref="idCard"></one-to-one>
	</class>
</hibernate-mapping>

添加了<one-to-one>标签,同时使用了property-ref属性,看下查询再解释:
@Test
	public void getPerson(){
		Session session=hibernateDao.getSession();
		Transaction tx=session.beginTransaction();
		
		IDCard idCard=(IDCard) session.get(IDCard.class,2L);
		System.out.println(idCard.getPerson().getName());
		
		tx.commit();
		session.close();
	}

查询的sql如下:
Hibernate: select idcard0_.id as id1_1_0_, idcard0_.number as number2_1_0_, idcard0_.content as content3_1_0_, person1_.id as id1_3_1_, person1_.name as name2_3_1_, person1_.age as age3_3_1_, person1_.idcard_id as idcard_i4_3_1_ from hibernate.idcard idcard0_ left outer join hibernate.person person1_ on idcard0_.id=person1_.idcard_id where idcard0_.id=?
张三

将Person查出来了,然后看下他是怎么查的,看下IDCard的<one-to-one name="person" property-ref="idCard">,name指的是IDCard的person属性,而property-ref指向了另一个类的属性,这里就是Person类的idCard属性,要想查出Person总要告诉hibernate idcard表的哪个字段和person表的哪个字段相连接吧,这里的property-ref="idCard",即指定了要和Person类的idCard属性所对应的字段相连接,默认是采用主键来和该字段相连接的,我不知道能否指定。即idcard表的主键id和Person类的idCard属性对应的字段idcard_id相连接,来查询Person,看sql语句:hibernate.idcard idcard0_ left outer join hibernate.person person1_ on idcard0_.id=person1_.idcard_id

上述一对一的关联关系是通过外键来连接,他们也可以使用相同的主键实现一对一。这里不再讲述。

若想转载请注明出处
作者:乒乓狂魔

© 著作权归作者所有

共有 人打赏支持
乒乓狂魔
粉丝 996
博文 105
码字总数 271356
作品 0
长宁
程序员
Hibernate系列——总结篇(九)

概念 Hibernate是一个对象关系映射框架,当然从分层的角度看,我们也说它是数据持久层的框架。 我们从上一句话可以看出Hibernate的核心:面向对象、关系映射以及数据持久化。前面两个概念很容...

architect刘源源
01/11
5
0
Hibernate Hibernate关联映射

Hibernate映射关系概述: Hibernate关联映射分为: ①、多对一。②、一对多。③、一对一。④、多对多。⑤、组件映射。⑥、集合映射。 在Uml语言中关联是有方向的,以客户Customer和订单Order...

Winnie007
2015/08/06
0
0
Hibernate映射——一对一双向关联映射(六)

我们介绍了一对一的单向关联映射,单向是指只能从人(Person)这端加载身份证端(IdCard),但是反过来,不能从身份证端加载人得信息。如图所示: 关键原因在于对象模型具有方向性: 单向:一...

architect刘源源
01/11
3
0
Hibernate(三)——框架中的关系映射

在设计数据库时我们会考虑,表与表之间的关系,例如我们前边经常提到的一对一,一对多,多对多关系,在数据库中我们通过外键,第三张表等来实现这些关系。而Hibernate时间实体类和数据库中的...

凡尘里的一根葱
2015/11/11
0
0
Hibernate映射——一对一单向关联映射(五)

映射原理 两个实体对象之间是一对一的关联映射,即一个对象只能与另外唯一的一个对象相对应。例如:一个人(Person)只有一张身份证(IdCard)。我们看一下这个例子的对象模型,如下图所示: 对象...

architect刘源源
01/11
2
0

没有更多内容

加载失败,请刷新页面

加载更多

Mac OS X下Maven的安装与配置

Mac OS X 安装Maven: 下载 Maven, 并解压到某个目录。例如/Users/robbie/apache-maven-3.3.3 打开Terminal,输入以下命令,设置Maven classpath $ vi ~/.bash_profile 添加下列两行代码,之后...

TonyStarkSir
今天
3
0
关于编程,你的练习是不是有效的?

最近由于工作及Solution项目的影响,我在重新学习DDD和领域建模的一些知识。然后,我突然就想到了这个问题,以及我是怎么做的? 对于我来说,提升技能的项目会有四种: 纯兴趣驱动的项目。即...

问题终结者
今天
3
0
打开eclipse出现an error has occurred see the log file

解决方法: 1,打开eclipse安装目录下的eclipse.ini文件; 2,打开的文本文件最后添加一行 --add-modules=ALL-SYSTEM 3,保存重新打开Eclipse。...

任梁荣
昨天
4
0
搞定Northwind示例数据库,无论哪个版本的SQLServer都受用

Northwind数据库 从这里可以找到突破口: http://social.msdn.microsoft.com/Forums/zh-CN/Vsexpressvb/thread/8490a1c6-9018-40c9-aafb-df9f79d29cde 下面是MSDN: http://msdn2.microsoft......

QQZZFT
昨天
1
0
mysql主从同步,安装配置操作

准备 两台mysql服务,我这里准备了如下: 主库:192.168.176.128 从库:192.168.176.131 如何在Linux上安装mysql服务,请看https://blog.csdn.net/qq_18860653/article/details/80250499 操作...

小致dad
昨天
5
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部