A:先讲讲一对一的关系(欲知其他关系,请看下篇)

a:主键关联的一对一关系

一对一关系一般用主键关联,也就是说用主键值来维护两者的关系,一个表的主键存放另一个表的主键值。例如在员工与帐号中,我们取员工表的主键值作为帐号的主键值。

我们一员工表和账号表为例:(员工表是主表,账号表是从表)

对持久化的对象的封装和get,set方法已省略,值得注意的是:vo中必须相互写上对方的对象:如在employee中要定义private AccountVo account,在account中也要写上对应的employee,   我们只对映射文件与测试编写

主表的mapping配置:使用了one-to-one

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="com.ysq.vo">
<class name="EmployeeVo" table="employee">
<id name="oid">
<generator class="sequence">
<param name="sequence">dept_seq</param> <!-- 使用的Oracle数据库,主键增长方式是sequence-->
</generator>
</id>
<property name="deptid">
<column name="deptid"/>
</property>
<property name="empName">
<column name="empName" length="20"/>
</property>
<property name="sex">
<column name="sex" length="2"/>
</property>
<property name="birthday">
<column name="birthday" length="30"/>
</property>
<property name="school">
<column name="school" length="20"/>
</property>
<property name="major">
<column name="major" length="10"/>
</property>
<property name="degree">
<column name="degree"/>
</property>
<property name="phone">
<column name="phone" length="12"/>
</property> <one-to-one name="account" class="AccountVo" cascade="all"></one-to-one>
</class>
</hibernate-mapping>

从表的mapping配置:

 <?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="com.ysq.vo">
<class name="AccountVo" table="account">
<id name="oid" type="java.lang.Integer">
<column name="oid" />
<generator class="foreign" ><!-- 采用外键方式生成主键值 -->
<param name="property">employee</param><!-- 表示取员工的主键值作为帐号的主键值,这里的employee要跟下面的<one-to-one>的name属性值一致 -->
</generator>
</id>
<property name="username" type="java.lang.String">
<column name="username" length="20" />
</property>
<property name="password" type="java.lang.String">
<column name="password" length="20" />
</property>
<property name="email" type="java.lang.String">
<column name="email" length="100" />
</property>
<property name="inactive" type="java.lang.String">
<column name="inactive" length="1" />
</property>
<one-to-one name="employee" class= "EmployeeVo" constrained="true"></one-to-one>
</class>
</hibernate-mapping>

对应的测试类:

 //添加员工信息,同时也添加了对应的账号
@Test
public void addEmployee(){
EmployeeVo employee = new EmployeeVo();
AccountVo account = new AccountVo(); account.setUsername("zhangsan21");
account.setEmail("csw-java@163.com");
account.setPassword("123"); employee.setBirthday("2003-09-09");
employee.setEmpName("lizi21");
employee.setPhone("1232132"); Session session = SessionFactoryUtils.getSession();
Transaction tr = session.beginTransaction();
try {
tr.begin();
/* //保存员工
session.save(employee);
//保存帐号
account.setEmployee(employee);
session.save(account);*/ //也可以进行双向关联,对主表进行save()
account.setEmployee(employee);
employee.setAccount(account);
session.save(employee);//保存员工时,级联保存了账号
tr.commit();
} catch (HibernateException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}finally{
session.close();
} }
@Test    //修改和删除
public void updateEmp(){
.......
try {
tr.begin();
//设置了双向关联,当对员工修改时,对account也进行了修改
employee.setAccount(account);
account.setEmployee(employee);
session.update(employee);
tr.commit();
} catch (HibernateException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}finally{
session.close();
} }
@Test
public void deleteEmp(){
........
Session session = SessionFactoryUtils.getSession();
Transaction tr = session.beginTransaction();
try {
tr.begin();
//设置双向关联
employee.setAccount(account);
account.setEmployee(employee);
 
 
               session.delete(employee);
tr.commit();
} catch (HibernateException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}finally{
session.close();
}
}
@Test   //查询信息
public void findEmployeeById(){
Session session = SessionFactoryUtils.getSession();
//一对一默认的延迟加载 get(class clazz,id)只针对通过id查询
EmployeeVo employee = (EmployeeVo)session.get(EmployeeVo.class, 6); session.close(); System.out.println(employee);
System.out.println(employee.getAccount().getUsername());//注意:我这里的打印是引用了vo中的toString()方法,但只能在一个封装类中写toString()方法,同时写两个会报错的
} @SuppressWarnings("unchecked")
@Test
public void findEmployee(){
Session session = SessionFactoryUtils.getSession();
//一对一默认的延迟加载[HQL]
List<EmployeeVo> emps = session.createQuery("from EmployeeVo").list(); session.close();
for (EmployeeVo employee2 : emps) {
System.out.println(employee2);
System.out.println(employee2.getAccount().getUsername());
System.out.println();
}
}

b:使用外键关联一对一的关系

只需要包对应的mapping文件中修改一些就可以了

如:employee中的:<one-to-one name="account" class="AccountVo" cascade="all"></one-to-one>需要修改如下:

<!--因为现在员工对帐号是采用外键关联,所以在这里得加一个属性property-ref="employee"指定对方many-to-one 的name属性值-->
     <one-to-one name="account" class="AccountVo" property-ref="employee" cascade="all"></one-to-one>

account中的:<one-to-one name="employee" class= "EmployeeVo" constrained="true"></one-to-one>  需要修改如下:

<many-to-one name="employee" class="EmployeeVo" column="empid" unique="true"></many-to-one><!--在这边不再配置one-to-one了,而是用many-to-one,并且在many-to-one中加了unique="true"属性,表示唯一的多对一,也就成了一对一了,这里还得注意必须得指定外键column="empid" -->

Hibernate中的一对一关系详解(1)的更多相关文章

  1. 分享知识-快乐自己:Hibernate 中Criteria Query查询详解

    1):Hibernate 中Criteria Query查询详解 当查询数据时,人们往往需要设置查询条件.在SQL或HQL语句中,查询条件常常放在where子句中. 此外,Hibernate还支持Cr ...

  2. Hibernate中的事务处理流程详解

    一.Hibernate操作的基本流程 使用 Hibernate 进行数据持久化操作,通常有如下步骤: 1.编写持久化类: POJO + 映射文件 2.获取 Configuration 对象 3.获取 ...

  3. Hibernate 中Criteria Query查询详解【转】

    当查询数据时,人们往往需要设置查询条件.在SQL或HQL语句中,查询条件常常放在where子句中.此外,Hibernate还支持Criteria查询(Criteria Query),这种查询方式把查询 ...

  4. Hibernate中的多对多关系详解(3)​

    前面两节我们讲到了一对一的关系,一对多,多对一的关系,相对来说,是比较简单的,但有时,我们也会遇到多对多的关系,比如说:角色与权限的关系,就是典型的多对多的关系,因此,我有必要对这种关系详解,以便大家 ...

  5. Hibernate中的一对一映射关系

    Hibernate中的一对一映射关系有两种实现方法(单向一对一,和双向一对一)(一对一关系:例如一个department只能有一个manager) 单向和双向有什么区别呢??例如若是单向一对一,比如在 ...

  6. Hibernate配置文件和映射文件详解

    Hibernate是一个彻底的ORM(Object Relational Mapping,对象关系映射)开源框架. 我们先看一下官方文档所给出的,Hibernate 体系结构的高层视图: 其中PO=P ...

  7. slf4j log4j logback关系详解和相关用法

    slf4j log4j logback关系详解和相关用法 写java也有一段时间了,一直都有用slf4j log4j输出日志的习惯.但是始终都是抱着"拿来主义"的态度,复制粘贴下配 ...

  8. c++中vector的用法详解

    c++中vector的用法详解 vector(向量): C++中的一种数据结构,确切的说是一个类.它相当于一个动态的数组,当程序员无法知道自己需要的数组的规模多大时,用其来解决问题可以达到最大节约空间 ...

  9. 【转】UML类图与类的关系详解

    UML类图与类的关系详解   2011-04-21 来源:网络   在画类图的时候,理清类和类之间的关系是重点.类的关系有泛化(Generalization).实现(Realization).依赖(D ...

随机推荐

  1. ZBar只扫描二维码/条形码

    You can add these codes for ImageScanner scanner.setConfig(0, Config.ENABLE, 0); //Disable all the S ...

  2. BABOK - 企业分析(Enterprise Analysis)

    BABOK - 企业分析(Enterprise Analysis)概要 发表于2013年10月9日由周金根 描述 企业分析描述我们如何捕捉.提炼并明晰业务需要,并定义一个可能实现这些业务需要的一个方案 ...

  3. 【ThinkingInC++】65、使用delete void*可能会出错

    /** * 书本:[ThinkingInC++] * 功能:使用delete void*可能会出错 * 时间:2014年10月5日14:31:43 * 作者:cutter_point */ #incl ...

  4. [转]flume-ng+Kafka+Storm+HDFS 实时系统搭建

    http://blog.csdn.net/weijonathan/article/details/18301321 一直以来都想接触Storm实时计算这块的东西,最近在群里看到上海一哥们罗宝写的Flu ...

  5. Android开发_后台任务task管理_allowTaskReparenting alwaysRetainTaskState clearTaskOn

    1.android:allowTaskReparenting 这个属性用来标记一个Activity实例在当前应用退居后台后,是否能从启动它的那个task移动到有共同affinity的task,“tru ...

  6. java反射性能

    项目中用到了java的反射,可以大大减少代码量.但是反射的性能却不容乐观,做了个简单的测试,如下. public void noreflect() { Person p = new Person(); ...

  7. Java的浮点数

    为了防止无良网站的爬虫抓取文章,特此标识,转载请注明文章出处.LaplaceDemon/ShiJiaqi. http://www.cnblogs.com/shijiaqi1066/p/5160771. ...

  8. codeforces 505C Mr. Kitayuta, the Treasure Hunter(dp)

    题意:有30001个岛,在一条线上,从左到右编号一次为0到30000.某些岛屿上有些宝石.初始的时候有个人在岛屿0,他将跳到岛屿d,他跳跃的距离为d.如果当前他跳跃的距离为L,他下一次跳跃的距离只能为 ...

  9. PHP获得header头进行分析

    学web的人都知道,要深刻的理解就一定要对HTTP协议有深刻的理解,这样你才能理解整个运行的流程,有些功能你才能理解应该 如何去实现,比如:仿盗链啊,定义IP后切换页面语种的版本啊,等等, 这里就来对 ...

  10. C#设置与获取目录权限(.net控制ACL)

    找到两种方式可以修改文件夹的权限 第一种: 想用c#来设置和读取ntfs分区上的目录权限,找了很多资料,未果.终于发现了一段vb.net的代码,做了修改,以C#展示给大家. using System; ...