举例如下

Customer类:

  1. public class Customer {
  2. private int id;
  3. private String name;
  4. private Set orders = new HashSet();
  5. •••
  6. }

即Customer类具有一个set集合属性orders,其中Order是一个普通的类:

  1. public class Order {
  2. private int id;
  3. private String orderName;
  4. •••
  5. }

数据库中表的结构:

  1. t_customer:  两个字段:id  name
  2. t_order:     三个字段:id  orderName  customerid

Customer类的映射文件:Customer.hbm.xml  (Order类的映射文件忽略)

  1. <hibernate-mapping>
  2. <class name="test.Customer" table="t_customer" lazy="false">
  3. <id name="id">
  4. <generator class="native"/>
  5. </id>
  6. <property name="name"/>
  7. <set name="orders"  cascade="save-update"  lazy="false">
  8. <key column="customerid"/>
  9. <one-to-many class="test.Order"/>
  10. </set>
  11. </class>
  12. </hibernate-mapping>

执行如下代码:

  1. Set orders = new HashSet();
  2. Order o1 = new Order();
  3. o1.setOrderName("o1");
  4. Order o2 = new Order();
  5. o2.setOrderName("o2");
  6. orders.add(o1);
  7. orders.add(o2);
  8. Customer c = new Customer();
  9. c.setName("aaa");
  10. c.setOrders(orders);
  11. session.save(c);

此时Hibernate发出的sql语句如下:

  1. Hibernate: insert into t_customer (name) values (?)
  2. Hibernate: insert into t_order (orderName) values (?)
  3. Hibernate: insert into t_order (orderName) values (?)
  4. Hibernate: update t_order set customerid=? where id=?
  5. Hibernate: update t_order set customerid=? where id=?

查看数据库:

  1. t_customer :                    t_order:
  2. id   |  name                   id   |   orderName   |   customerid
  3. 1       aaa                    1           o1              1
  4. 2           o2              1

保存Customer对象时,首先发出insert into t_customer (name) values (?)语句将c同步到数据库,由于在<set>映射中设置cascade="save-update",所以会同时保存orders集合中的Order类型的o1,o2对象(如果没有这个设置,即cascade="save-update"),那么Hibenrate不会自动保存orders集合中的对象,那么在更新时将会抛出如下异常:

  1. Hibernate: insert into t_customer (name) values (?)
  2. Hibernate: update t_order set customerid=? where id=?
  3. org.hibernate.TransientObjectException: test.Order
  4. ••••••

抛出这一异常的原因是:<set>映射默认"inverse=fasle"即由Customer对象作为主控方,那么它要负责关联的维护工作,在这里也就是负责更新t_order表中的customerid字段的值,但由于未设置cascade="save-update",所以orders集合中的对象不会在保存customer时自动保存,因此会抛出异常(如果未设置,需要手动保存)。 
现在设置cascade="save-update",同时设置inverse="true",即:

  1. •••
  2. <set name="orders" cascade="save-update" inverse="true" lazy="false">
  3. <key column="customerid"/>
  4. <one-to-many class="test.Order"/>
  5. </set>
  6. •••

同样执行上述代码,发出如下语句:

  1. Hibernate: insert into t_customer (name) values (?)
  2. Hibernate: insert into t_order (orderName) values (?)
  3. Hibernate: insert into t_order (orderName) values (?)

相比上一次执行,少了两条update语句,查看数据库:

  1. t_customer :                    t_order:
  2. id   |  name                   id   |   orderName   |   customerid
  3. 1       aaa                    1           o1              NULL
  4. 2           o2              NULL

发现t_order表中customerid的值为NULL,这是由于设置了inverse="true",它意味着 
Customer不再作为主控方,而将关联关系的维护工作交给关联对象Orders来完成。在保存Customer时,Customer不在关心Orders的customerid属性,必须由Order自己去维护,即设置order.setCustomer(customer);

如果需要通过Order来维护关联关系,那么这个关联关系转换成双向关联。 
修改Order类代码:

  1. public class Order {
  2. private int id;
  3. private String orderName;
  4. private Customer customer;
  5. •••
  6. }

Order.hbm.xml:

  1. <hibernate-mapping>
  2. <class name="test.Order" table="t_order">
  3. <id name="id">
  4. <generator class="native"/>
  5. </id>
  6. <property name="orderName"/>
  7. <many-to-one name="customer" column="customerid"/>
  8. </class>
  9. </hibernate-mapping>

此时数据库中表的结构不会变化。

再次执行上述代码,发出如下sql语句:

  1. Hibernate: insert into t_customer (name) values (?)
  2. Hibernate: insert into t_order (orderName, customerid) values (?, ?)
  3. Hibernate: insert into t_order (orderName, customerid) values (?, ?)

发现在保存Order对象时为customerid字段赋值,因为Order对象中拥有Customer属性,对应customerid字段,查看数据库表:

  1. t_customer :                    t_order:
  2. id   |  name                   id   |   orderName   |   customerid
  3. 1       aaa                    1           o1              NULL
  4. 2           o2              NULL

发现customerid的值仍为NULL,因为在上述代码中并未设置Order对象的Customer属性值,由于设置了inverse="true",所以Order对象需要维护关联关系,所以必须进行设置,即 
order.setCustomer(customer);

修改上述代码为:

  1. •••
  2. Customer c = new Customer();
  3. Set orders = new HashSet();
  4. Order o1 = new Order();
  5. o1.setOrderName("o1");
  6. o1.setCustomer(c);
  7. Order o2 = new Order();
  8. o2.setOrderName("o2");
  9. o2.setCustomer(c);
  10. orders.add(o1);
  11. orders.add(o2);
  12. c.setName("aaa");
  13. c.setOrders(orders);
  14. session.save(c);
  15. •••

执行上述代码,发出如下语句:

  1. Hibernate: insert into t_customer (name) values (?)
  2. Hibernate: insert into t_order (orderName, customerid) values (?, ?)
  3. Hibernate: insert into t_order (orderName, customerid) values (?, ?)

查看数据库:

  1. t_customer :                    t_order:
  2. id   |  name                   id   |   orderName   |   customerid
  3. 1       aaa                    1           o1              1
  4. 2           o2              1

发现已经设置了customerid的值。

在一对多关联中,在一的一方设置inverse="true"(即由多来维护!!),有助于性能的改善。通过上述分析可以发现少了update语句。

hibernate 一对多,由谁维护性能最优的更多相关文章

  1. hibernate 一对多双向关联 详解

    一.解析: 1.  一对多双向关联也就是说,在加载班级时,能够知道这个班级所有的学生. 同时,在加载学生时,也能够知道这个学生所在的班级. 2.我们知道,一对多关联映射和多对一关联映射是一样的,都是在 ...

  2. Hibernate一对多OnetoMany

    ------------------------Hibernate一对多OnetoMany 要点: 配置在一端. 1.如果是单向关联,即只在一端配置OneToMany,多端不配置ManyToOne.则 ...

  3. Hibernate一对多单向关联和双向关联映射方法及其优缺点 (待续)

    一对多关联映射和多对一关联映射实现的基本原理都是一样的,既是在多的一端加入一个外键指向一的一端外键,而主要的区别就是维护端不同.它们的区别在于维护的关系不同: 一对多关联映射是指在加载一的一端数据的同 ...

  4. Hibernate一对多操作

    --------------------siwuxie095 Hibernate 一对多操作 以客户和联系人为例,客户是一,联系人是多 即 一个客户里面有多个联系人,一个联系人只能属于一个客户 注意: ...

  5. Hibernate一对多自关联、多对多关联

    今天分享hibernate框架的两个关联关系    多对多关系注意事项 一定要定义一个主控方 多对多删除 主控方直接删除 被控方先通过主控方解除多对多关系,再删除被控方 禁用级联删除 关联关系编辑,不 ...

  6. Java进阶知识10 Hibernate一对多_多对一双向关联(Annotation+XML实现)

    本文知识点(目录): 1.Annotation 注解版(只是测试建表)    2.XML版 的实现(只是测试建表)    3.附录(Annotation 注解版CRUD操作)[注解版有个问题:插入值时 ...

  7. hibernate一对多关联映射

    一对多关联映射 映射原理 一对多关联映射和多对一关联映射的映射原理是一致的,都是在多的一端加入一个外键,指向一的一端.关联关系都是由多端维护,只是在写映射时发生了变化. 多对一和一对多的区别 多对一和 ...

  8. Hibernate一对多配置

    刚刚学习了Hibernate框架的基础知识,下面我来说说关于Hibernate一对多的配置 首先是大配置 连接数据库 用户名 和密码 能和小配置连接 部门小配置: 员工小配置: 部门实体类 员工实体类 ...

  9. Spring / Hibernate 应用性能调优

    来源:ImportNew - 陈晓舜 对大部分典型的Spring/Hibernate企业应用来说,应用的性能大部分由持久层的性能决定. 这篇文章会重温一下怎么去确认我们的应用是否是”数据库依赖(dat ...

随机推荐

  1. 正则表达式在JS中的使用

    <script type="text/javascript"> /** *正则表达式在js中的第一种使用方式: * RegExp 通过构造器去使用正则表达式 需要对反斜 ...

  2. python的map函数和reduce函数(转)

    map函数 map()函数 map()是 Python 内置的高阶函数,它接收一个函数 f 和一个 list,并通过把函数 f 依次作用在 list 的每个元素上,得到一个新的 list 并返回. 例 ...

  3. XtraBackup 备份原理

    来着淘宝技术: http://mysql.taobao.org/monthly/2016/03/07/ https://github.com/alibaba/AliSQL 前言 Percona Xtr ...

  4. bedtools简介及应用

    1)背景处理基因组数据中,比较基因组不同区域,例如寻找overlap等,是一种基本的且常见的问题.虽然UCSC 中‘Table Browser’或者Galaxy可以用来处理,但是当这些工具面对大的数据 ...

  5. Maven(一)——Maven入门

    一.Maven的基本概念 Maven(翻译为"专家","内行")是跨平台的项目管理工具.主要服务于基于Java平台的项目构建,依赖管理和项目信息管理. 1.1. ...

  6. 安装使用phpStudy在本机配置php运行环境

    前言: php开发的初学者,强烈推荐使用phpStudy集成环境,一方面这个的确很好用(本人电脑安装了jspStudy,可以同时调试php和jsp),另一方面呢,虽然本人是技术控,但对这些繁杂的安装部 ...

  7. TZOJ 3533 黑白图像(广搜)

    描述 输入一个n*n的黑白图像(1表示黑色,0表示白色),任务是统计其中八连块的个数.如果两个黑格子有公共边或者公共顶点,就说它们属于同一个八连块.如图所示的图形有3个八连块. 输入 第1行输入一个正 ...

  8. 300最长上升子序列 · Longest Increasing Subsequence

    [抄题]: 往上走台阶 最长上升子序列问题是在一个无序的给定序列中找到一个尽可能长的由低到高排列的子序列,这种子序列不一定是连续的或者唯一的. 样例 给出 [5,4,1,2,3],LIS 是 [1,2 ...

  9. 349. Intersection of Two Arrays 是否重复

    不重复的: 方法一 [一句话思路]:排序之后用归并. [一刷]: 根据返回方法的类型来判断corner case.判断空.0数组的corner case还不一样,得分开写 由于先排序了,nums1[i ...

  10. php Pthread 多线程 (三) Mutex 互斥量

    当我们用多线程操作同一个资源时,在同一时间内只能有一个线程能够对资源进行操作,这时就需要用到互斥量了.比如我们对同一个文件进行读写操作时. <?php class Add extends Thr ...