创建JavaBean

一方: Customer

private long cust_id;
private String cust_name;
private long cust_user_id;
private long cust_create_id;
private String cust_source;
private String cust_industry;
private String cust_level;
private String cust_linkman;
private String cust_phone;
private String cust_mobile; //存储联系人的集合,该属性并不会出现在数据库中
private Set<LinkMan> linkmans = new HashSet<LinkMan>(); //--加上所有的get/set方法 无参构造函数

多方: LinkMan

private long lkm_id;
private String lkm_name;
private String lkm_gender;
private String lkm_phone;
private String lkm_mobile;
private String lkm_email;
private String lkm_qq;
private String lkm_position;
private String lkm_memo; //外键对象
private Customer customer; //--加上所有的get/set方法 无参构造函数

配置多方的映射文件

<!-- 先配置多方
name:当前JavaBean中的属性
class:属性的全路径
column:外键的字段
-->
<many-to-one name="customer" class="com.hibernateday3.domain.Customer" column="lkm_cust_id"/>

配置一方的映射文件

 <!-- 后配置一方
set->name:表示集合的名称
-->
<set name="linkmans">
<!-- 外键的字段与路径 -->
<key column="lkm_cust_id"/>
<one-to-many class="com.hibernateday3.domain.LinkMan"/>
</set>

将映射文件配置到核心文件

<!-- 映射配置文件 -->
<mapping resource="com/hibernateday3/domain/Customer.hbm.xml"/>
<mapping resource="com/hibernateday3/domain/Linkman.hbm.xml"/>

测试:

1.测试双向关联方式

@Test
public void m01(){
Session session = HibernateUtils.openSession();
Transaction tr = session.beginTransaction(); Customer c1 = new Customer();
c1.setCust_name("客户1");
LinkMan l1 = new LinkMan();
l1.setLkm_name("联系人1");
LinkMan l2 = new LinkMan();
l2.setLkm_name("联系人2"); c1.getLinkmans().add(l1);
c1.getLinkmans().add(l2); //------保存数据
//双向关联保存
session.save(c1);
session.save(l1);
session.save(l2); //------提交事务释放资源
tr.commit();
session.close();
}

2.单向的关联,如果不配置级联保存,程序出现异常

//------保存数据
//双向关联保存 session.save(c1); //------提交事务释放资源
警告:org.hibernate.TransientObjectException: object references an unsaved transient instance - save the transient instance before flushing: com.hibernateday3.domain.LinkMan

如果想完成只保存一方的数据,并且把相关联的数据都保存到数据库中,那么需要配置级联!!

级联保存是具有方向性的

3.测试级联保存:保存客户,级联联系人

---在一方映射文件加入cascade属性

<set name="linkmans" cascade="save-update">
<!-- 外键的字段与路径 -->
<key column="lkm_cust_id"/>
<one-to-many class="com.hibernateday3.domain.LinkMan" />
</set>
//------保存数据
//双向关联保存 session.save(c1); //------提交事务释放资源

运行成功。。

4.测试级联保存:保存联系人,级联客户

---在多方映射文件加入cascade属性

<many-to-one name="customer" class="com.hibernateday3.domain.Customer" column="lkm_cust_id" cascade="save-update"/>
//------保存数据
//双向关联保存 session.save(l1);
session.save(l2); //------提交事务释放资源

运行成功。。

5.测试级联保存


//------保存数据
//双向关联保存
l1.setCustomer(c1);
c1.getLinkmans().add(l2);
session.save(l1);
//------提交事务释放资源

运行成功。。

6.测试:删除一方,一方下有2个联系人

public void m06(){
Session session = HibernateUtils.openSession();
Transaction tr = session.beginTransaction(); //------保存数据
//双向关联保存 Customer c1 = session.get(Customer.class, 1L);
session.delete(c1); //------提交事务释放资源
tr.commit();
session.close();
}

能够成功 并不会因为一方是多方的外键二不能删除
Hibernate先让多方删除外键 再删除一方

7.测试级联删除,删除客户,级联删除客户下的联系人

在一方映射文件下添加属性

<set name="linkmans" cascade="save-update,delete">
<!-- 外键的字段与路径 -->
<key column="lkm_cust_id"/>
<one-to-many class="com.hibernateday3.domain.LinkMan" />
</set>
//------保存数据
//双向关联保存 Customer c1 = session.get(Customer.class, 1L);
session.delete(c1); //------提交事务释放资源

删除c1  c1对应的l1,l2都被删除了

8.删除联系人,级联删除客户

在多方映射文件下添加属性

    <many-to-one name="customer" class="com.hibernateday3.domain.Customer" column="lkm_cust_id" cascade="save-update,delete"/>
//------保存数据
//双向关联保存 LinkMan l1 = session.get(LinkMan.class,1L ); session.delete(l1); //------提交事务释放资源

如果一方下的cascade的delete属性依然存在,所有的数据都被删除

9.解除关系,从一方集合中移除多方数据

// 先获取到客户
Customer c1 = session.get(Customer.class, 1L);
LinkMan l1 = session.get(LinkMan.class, 1L);
// 解除
c1.getLinkmans().remove(l1);

多方的外键被解除设为null

如果cascade属性为delete-orphan 解除关系 ,多方记录将被删除(孤儿关系)

1. 取值如下
* none -- 不使用级联
* save-update -- 级联保存或更新
* delete -- 级联删除
* delete-orphan -- 孤儿删除.(注意:只能应用在一对多关系)
* all -- 除了delete-orphan的所有情况.(包含save-update delete)
* all-delete-orphan -- 包含了delete-orphan的所有情况.(包含save-update delete delete-orphan) 2. 孤儿删除(孤子删除),只有在一对多的环境下才有孤儿删除
* 在一对多的关系中,可以将一的一方认为是父方.将多的一方认为是子方.孤儿删除:在解除了父子关系的时候.将子方记录就直接删除。
* <many-to-one cascade="delete-orphan" />

放弃外键维护

1. 先测试双方都维护外键的时候,会产生多余的SQL语句。
* 想修改客户和联系人的关系,进行双向关联,双方都会维护外键,会产生多余的SQL语句。
/**
* 放弃外键的维护
* 需求:让c1联系人l1属于c2
*/
@Test
public void m11(){
Session session = HibernateUtils.getCurrentSession();
Transaction tr = session.beginTransaction(); Customer c2 = session.get(Customer.class,2L);
LinkMan l1 = session.get(LinkMan.class, 1L); c2.getLinkmans().add(l1);
l1.setCustomer(c2); tr.commit();
}


    * 产生的原因:session的一级缓存中的快照机制,会让双方都更新数据库,产生了多余的SQL语句。

2. 如果不想产生多余的SQL语句,那么需要一方来放弃外键的维护!
* 在<set>标签上配置一个inverse=”true”.true:放弃.false:不放弃.默认值是false
* <inverse="true">
<set name="linkmans" inverse="true">
<!-- 外键的字段与路径 -->
<key column="lkm_cust_id"/>
<one-to-many class="com.hibernateday3.domain.LinkMan" />
</set>
/**
* 放弃外键的维护
* 需求:让c1联系人l1属于c2
*/
@Test
public void m11(){
Session session = HibernateUtils.getCurrentSession();
Transaction tr = session.beginTransaction(); Customer c2 = session.get(Customer.class,2L);
LinkMan l1 = session.get(LinkMan.class, 1L); c2.getLinkmans().add(l1);
l1.setCustomer(c2); tr.commit();
}


cascade和inverse的区别
1. cascade用来级联操作(保存、修改和删除)---配置在多方
2. inverse用来维护外键的  ----配置在一方

11.Hibernate一对多关系的更多相关文章

  1. 菜鸟学习Hibernate——一对多关系映射

    Hibernate中的关系映射,最常见的关系映射之一就是一对多关系映射例如学生与班级的关系,一个班级对应多个学生.如图: Hibernate中如何来映射这两个的关系呢? 下面就为大家讲解一下: 1.创 ...

  2. hibernate一对多关系配置

    一.     表信息 公司表 cId cName cAdress Null Null Null 表t_company 员工表 sId sName sAge cId Null Null Null Nul ...

  3. Hibernate一对多关系操作

    1.创建两个实体类. 一个实体类是商品类,另一个实体类是商品的分类类. 在一对多关系的两个实体中,在编写实体类时必须要遵循以下规则: (1)在一的那一方的实体中,必须要有一个私有的多那一方的实体对象属 ...

  4. hibernate 一对多关系中的孤儿属性

    @OneToMany(targetEntity = BenefitType.class, mappedBy = "sitePerson",cascade = CascadeType ...

  5. hibernate一对多关系映射(自身关联)

    示例:一个类别(Category)下面有多个子类别,多个子类别属于同一个父类别. public class Category  { private Integer id; private String ...

  6. hibernate一对多关系 在一方查询会获得重复数据,重复数量就是多端数据数量用@Fetch(FetchMode.SUBSELECT)解决

    先来看数据表 版块表只有两个数据 板块1是推荐,下边没有子栏目 板块2下边有14个子栏目 在1的一端来查询,发现结果有16条 也就是板块1+版块2+版块2和他的14个子集都列出来了,这明显不对 板块对 ...

  7. hibernate中一对多关系中的inverse,cascade属性

    举例说明: 一对多关系的两张表:boy.girl(一个男孩可以多个女朋友) boy表结构 Field   Type        ------  -----------  name    varcha ...

  8. 【Java EE 学习 45】【Hibernate学习第二天】【对象的三种状态】【一对多关系的操作】

    一.对象的三种状态. 1.对象有三种状态:持久化状态.临时状态.脱管状态(游离状态) 2.Session的特定方法能使得一个对象从一个状态转换到另外一个状态. 3.三种状态的说明 (1)临时状态:临时 ...

  9. hibernate学习(5)——一对多关系表达

    一对多实现 1. 实现类 package com.alice.hibernate02.vo; import java.util.HashSet; import java.util.Set; publi ...

随机推荐

  1. 记一次数据丢失(电脑硬盘closed to down)的经历

    早上-高高兴兴上班去. 到了公司,突然发现出现windows.logo一两秒的时候会蓝屏,surprise. 百度了一下代码,ok修改硬盘格式,从ACHI到IDE 进入Bios,嗯?感觉界面变了,咋回 ...

  2. case in

    #!/bin/bash source /etc/profilesource ~/.bashrc #自己定义$version_number case $version_number in3.0.17) ...

  3. 阿里云POLARDB如何帮助猿辅导打造“孩子喜欢老师好”的网课平台?

    海量的题库.音视频答题资料.用户数据以及日志,对猿辅导后台数据存储和处理能力都提出了严峻的要求.而由于教育辅导行业的业务特点,猿辅导也面临着业务峰值对于数据库能力的巨大挑战.本文就为大家介绍阿里云PO ...

  4. thinkphp 比较标签

    比较标签用于简单的变量比较,复杂的判断条件可以用if标签替换,比较标签是一组标签的集合,基本上用法都一致,如下: <比较标签 name="变量" value="值& ...

  5. spring源码读书笔记

    如果我们在web项目里面使用spring的话,通常会在web.xml里面配置一个listener. <listener> <listener-class> org.spring ...

  6. SpringBoot--springboot启动类和controller的配置

    作为一个springboot初学者,在探索过程中难免遇到一些坑,边看书边动手,发现书本中的版本是1.0,而我使用的是最新版2.0,所以有些东西不能完全按照书本进行操作,因为2.0中已经不支持1.0中的 ...

  7. 根据Cron表达式,通过Spring自带的CronSequenceGenerator类获取下次执行时间

    Cron表达式通常用于执行一些定时任务,在本篇文章中,暂时不会记录如何根据Cron表达式来执行一些定时任务.本章主要的目的是根据Cron表达式,通过Spring自带的CronSequenceGener ...

  8. ie9 jscript7 内存不足 页面无响应

    花了我差不多一天时间 我是加载一个datagrid ,多表联查,查询几遍(不一定,又是1遍就死了)后 就卡死了...后台日志都是过的.... 后来我发现数据库某个表的数据很多有一模一样的两条,把一份删 ...

  9. neo4j安装APOC插件

    1.APOC下载地址:https://github.com/neo4j-contrib/neo4j-apoc-procedures/releases/3.4.0.1 只要下载.jar这一个压缩文件就好 ...

  10. memcache 使用手册

    Memcached 教程 Memcached是一个自由开源的,高性能,分布式内存对象缓存系统. Memcached是以LiveJournal旗下Danga Interactive公司的Brad Fit ...