- 浏览: 218664 次
- 性别:
- 来自: 广州
文章分类
- 全部博客 (397)
- j2se (28)
- nio (3)
- 易错点 (3)
- 面试ssh (9)
- ssh整合 (11)
- jbpm+spring (2)
- js (15)
- 高级技术 (59)
- swing (3)
- 数据库 (16)
- hibernate (18)
- spring (19)
- 开发网站知识点 (9)
- jbpm (4)
- json (5)
- 设计模式 (22)
- 自定义标签 (1)
- j2ee (9)
- lucene (3)
- cahce (11)
- maven (5)
- html5 (1)
- 多数据源 (10)
- 页面聊天 (9)
- 富客户端 (1)
- android (13)
- aop+拦截器+jms (13)
- 框架整合 (1)
- 非阻塞io (24)
- 暂时不看 (13)
- webservice (3)
- oracle (3)
- 算法 (4)
- 协程 (2)
- netty (1)
- 爬虫 (0)
- 高级基础 (1)
- JVM调优总结 (12)
- 知识点技巧 (1)
- REST (0)
- 基础 io (2)
- dubbo (8)
- 线程 (1)
- spring源码 (2)
- git (1)
- office (2)
最新评论
-
sjzcmlt:
,写的挺好的啊
一个完整的负载均衡的例子 . -
他大姨妈:
网上大部分例子都是直接通过IdleStateHandler来实 ...
Netty的超时机制 心跳机制
一、Inverse是hibernate双向关系中的基本概念。inverse的真正作用就是指定由哪一方来维护之间的关联关系。当一方中指定了“inverse=false”(默认),那么那一方就有责任负责之间的关联关系,说白了就是hibernate如何生成Sql来维护关联的记录!
Hibernate仅仅按照主控方对象的状态的变化来同步更新数据库。按照原来的映射文件,people.getAddresses().add(address),即主控方对象的状态发生了改变,因此数据库会跟着对象状态的变化来同步更新数据库;而address.setPeople(people),即被控方对象的状态发生了改变,它是不能触发对象和数据库的同步更新的。
(实例1):
举个最简单的一对多父子关系。那么代码就写成:
[java] view plaincopyprint?
01.Parent p = new Parent();
02.Child c = new Child();
03.c.setParent(p); //维护父子之间关系
04.p.getChildren().add(c);
05.
06.session.save(p);
07.session.flush();
Parent p = new Parent();
Child c = new Child();
c.setParent(p); //维护父子之间关系
p.getChildren().add(c);
session.save(p);
session.flush(); 映射文件配置:
[html] view plaincopyprint?
01.父亲中的关系映射
02.{set name="children" lazy="true" inverse="true"}
03. {key column="parent_id"/}
04. {one-to-many class="test.Child"/}
05.{/set}
06.
07.儿子中关系映射
08.{many-to-one name="parent" column="parent_id" not-null="true"/}
父亲中的关系映射
{set name="children" lazy="true" inverse="true"}
{key column="parent_id"/}
{one-to-many class="test.Child"/}
{/set}
儿子中关系映射
{many-to-one name="parent" column="parent_id" not-null="true"/}
set中inverse="true",说明父子关系只在多的一端(Child)维护。所以只会发出2个insert语句。
注意:{many-to-one}总是设成“inverse=false”的,而且这个属性在Mapping中是不存在的!
这样运行的下来的结果就是:
[sql] view plaincopyprint?
01.Hibernate: insert into parent (id) values (?)
02.Hibernate: insert into child (parent_id, id) values (?, ?)
Hibernate: insert into parent (id) values (?)
Hibernate: insert into child (parent_id, id) values (?, ?) 如果将set中的inverse设为true,那么会发出3条sql语句,前2条是insert语句,后1条是update语句用来维护parent和child类的父子关系。
当然,假如c.setParent(p)注释掉(破坏了父子关系),结果就变成了:
[sql] view plaincopyprint?
01.Hibernate: insert into parent (id) values (?)
Hibernate: insert into parent (id) values (?)
===================================================
(实例2):
一个Person可以参加多个Event,一个Event有多个Person参加。映射文件如下:
[html] view plaincopyprint?
01.<!-- Person.hbm.xml -->
02.<hibernate-mapping package="events">
03. <class name="Person" table="person">
04. <id name="id" column="person_id">
05. <generator class="native"/>
06. </id>
07. <property name="age" length="0"/>
08. <property name="firstname"/>
09. <property name="lastname"/>
10. <set name="events" table="person_event">
11. <key column="person_id"/>
12. <many-to-many column="event_id" class="events.Event"/>
13. </set>
14. </class>
15.</hibernate-mapping>
16.
17.<!-- Event.hbm.xml -->
18.<hibernate-mapping>
19. <class name="events.Event" table="events">
20. <id name="id" column="event_id">
21. <generator class="native"/>
22. </id>
23. <property name="date" column="events_date" type="timestamp"/>
24. <property name="title" column="events_title"/>
25. <set name="participants" table="person_event" inverse="true">
26. <key column="event_id"/>
27. <many-to-many column="person_id" class="events.Person"/>
28. </set>
29. </class>
30.</hibernate-mapping>
<!-- Person.hbm.xml -->
<hibernate-mapping package="events">
<class name="Person" table="person">
<id name="id" column="person_id">
<generator class="native"/>
</id>
<property name="age" length="0"/>
<property name="firstname"/>
<property name="lastname"/>
<set name="events" table="person_event">
<key column="person_id"/>
<many-to-many column="event_id" class="events.Event"/>
</set>
</class>
</hibernate-mapping>
<!-- Event.hbm.xml -->
<hibernate-mapping>
<class name="events.Event" table="events">
<id name="id" column="event_id">
<generator class="native"/>
</id>
<property name="date" column="events_date" type="timestamp"/>
<property name="title" column="events_title"/>
<set name="participants" table="person_event" inverse="true">
<key column="event_id"/>
<many-to-many column="person_id" class="events.Person"/>
</set>
</class>
</hibernate-mapping>
inverse=true的含义: 由双向关联另一方维护该关联,己方不维护该关联(只能进行查询操作)。在上述代码中,由Person方维护该<many-to-many>关系,示例代码如下(以向Person参与的Event中加入新的Event为例):
[java] view plaincopyprint?
01.Session session = HibernateUtil.getSessionFactory().getCurrentSession();
02. session.beginTransaction();
03. Person p = (Person) session.load(Person.class, personId);
04. Event e = (Event) session.load(Event.class, eventId);
05. p.getEvents().add(e);//执行该代码时,hibernate会向中间表 person_event中插入person_id和event_id记录,如果换成e.getParticipants().add(p)的话,该代码将不会被执行,即hibernate不会向表person_event中插入记录。
06. session.getTransaction().commit();
Session session = HibernateUtil.getSessionFactory().getCurrentSession();
session.beginTransaction();
Person p = (Person) session.load(Person.class, personId);
Event e = (Event) session.load(Event.class, eventId);
p.getEvents().add(e);//执行该代码时,hibernate会向中间表 person_event中插入person_id和event_id记录,如果换成e.getParticipants().add(p)的话,该代码将不会被执行,即hibernate不会向表person_event中插入记录。
session.getTransaction().commit();
要注意的一点:在双向关联的关系中,映射的column(和table)的值要一致(即要用相同的表名和列名),不然设置为inverse="true"的这方将失去这个双向关系,而变成了一个单向关联。
二、Inverse和Cascade的比较
Inverse:负责控制关系,默认为false,也就是关系的两端都能控制,但这样会造成一些问题,更新的时候会因为两端都控制关系,于是重复更新。一般来说有一端要设为true。
Cascade:负责控制关联对象的级联操作,包括更新、删除等,也就是说对一个对象进行更新、删除时,其它对象也受影响,比如我删除一个对象,那么跟它是多对一关系的对象也全部被删除。
举例说明区别:删除“一”那一端一个对象O的时候,如果“多”的那一端的Inverse设为true,则把“多”的那一端所有与O相关联的对象外键清空;如果“多”的那一端的Cascade设为Delete,则把“多”的那一端所有与O相关联的对象全部删除。
Hibernate仅仅按照主控方对象的状态的变化来同步更新数据库。按照原来的映射文件,people.getAddresses().add(address),即主控方对象的状态发生了改变,因此数据库会跟着对象状态的变化来同步更新数据库;而address.setPeople(people),即被控方对象的状态发生了改变,它是不能触发对象和数据库的同步更新的。
(实例1):
举个最简单的一对多父子关系。那么代码就写成:
[java] view plaincopyprint?
01.Parent p = new Parent();
02.Child c = new Child();
03.c.setParent(p); //维护父子之间关系
04.p.getChildren().add(c);
05.
06.session.save(p);
07.session.flush();
Parent p = new Parent();
Child c = new Child();
c.setParent(p); //维护父子之间关系
p.getChildren().add(c);
session.save(p);
session.flush(); 映射文件配置:
[html] view plaincopyprint?
01.父亲中的关系映射
02.{set name="children" lazy="true" inverse="true"}
03. {key column="parent_id"/}
04. {one-to-many class="test.Child"/}
05.{/set}
06.
07.儿子中关系映射
08.{many-to-one name="parent" column="parent_id" not-null="true"/}
父亲中的关系映射
{set name="children" lazy="true" inverse="true"}
{key column="parent_id"/}
{one-to-many class="test.Child"/}
{/set}
儿子中关系映射
{many-to-one name="parent" column="parent_id" not-null="true"/}
set中inverse="true",说明父子关系只在多的一端(Child)维护。所以只会发出2个insert语句。
注意:{many-to-one}总是设成“inverse=false”的,而且这个属性在Mapping中是不存在的!
这样运行的下来的结果就是:
[sql] view plaincopyprint?
01.Hibernate: insert into parent (id) values (?)
02.Hibernate: insert into child (parent_id, id) values (?, ?)
Hibernate: insert into parent (id) values (?)
Hibernate: insert into child (parent_id, id) values (?, ?) 如果将set中的inverse设为true,那么会发出3条sql语句,前2条是insert语句,后1条是update语句用来维护parent和child类的父子关系。
当然,假如c.setParent(p)注释掉(破坏了父子关系),结果就变成了:
[sql] view plaincopyprint?
01.Hibernate: insert into parent (id) values (?)
Hibernate: insert into parent (id) values (?)
===================================================
(实例2):
一个Person可以参加多个Event,一个Event有多个Person参加。映射文件如下:
[html] view plaincopyprint?
01.<!-- Person.hbm.xml -->
02.<hibernate-mapping package="events">
03. <class name="Person" table="person">
04. <id name="id" column="person_id">
05. <generator class="native"/>
06. </id>
07. <property name="age" length="0"/>
08. <property name="firstname"/>
09. <property name="lastname"/>
10. <set name="events" table="person_event">
11. <key column="person_id"/>
12. <many-to-many column="event_id" class="events.Event"/>
13. </set>
14. </class>
15.</hibernate-mapping>
16.
17.<!-- Event.hbm.xml -->
18.<hibernate-mapping>
19. <class name="events.Event" table="events">
20. <id name="id" column="event_id">
21. <generator class="native"/>
22. </id>
23. <property name="date" column="events_date" type="timestamp"/>
24. <property name="title" column="events_title"/>
25. <set name="participants" table="person_event" inverse="true">
26. <key column="event_id"/>
27. <many-to-many column="person_id" class="events.Person"/>
28. </set>
29. </class>
30.</hibernate-mapping>
<!-- Person.hbm.xml -->
<hibernate-mapping package="events">
<class name="Person" table="person">
<id name="id" column="person_id">
<generator class="native"/>
</id>
<property name="age" length="0"/>
<property name="firstname"/>
<property name="lastname"/>
<set name="events" table="person_event">
<key column="person_id"/>
<many-to-many column="event_id" class="events.Event"/>
</set>
</class>
</hibernate-mapping>
<!-- Event.hbm.xml -->
<hibernate-mapping>
<class name="events.Event" table="events">
<id name="id" column="event_id">
<generator class="native"/>
</id>
<property name="date" column="events_date" type="timestamp"/>
<property name="title" column="events_title"/>
<set name="participants" table="person_event" inverse="true">
<key column="event_id"/>
<many-to-many column="person_id" class="events.Person"/>
</set>
</class>
</hibernate-mapping>
inverse=true的含义: 由双向关联另一方维护该关联,己方不维护该关联(只能进行查询操作)。在上述代码中,由Person方维护该<many-to-many>关系,示例代码如下(以向Person参与的Event中加入新的Event为例):
[java] view plaincopyprint?
01.Session session = HibernateUtil.getSessionFactory().getCurrentSession();
02. session.beginTransaction();
03. Person p = (Person) session.load(Person.class, personId);
04. Event e = (Event) session.load(Event.class, eventId);
05. p.getEvents().add(e);//执行该代码时,hibernate会向中间表 person_event中插入person_id和event_id记录,如果换成e.getParticipants().add(p)的话,该代码将不会被执行,即hibernate不会向表person_event中插入记录。
06. session.getTransaction().commit();
Session session = HibernateUtil.getSessionFactory().getCurrentSession();
session.beginTransaction();
Person p = (Person) session.load(Person.class, personId);
Event e = (Event) session.load(Event.class, eventId);
p.getEvents().add(e);//执行该代码时,hibernate会向中间表 person_event中插入person_id和event_id记录,如果换成e.getParticipants().add(p)的话,该代码将不会被执行,即hibernate不会向表person_event中插入记录。
session.getTransaction().commit();
要注意的一点:在双向关联的关系中,映射的column(和table)的值要一致(即要用相同的表名和列名),不然设置为inverse="true"的这方将失去这个双向关系,而变成了一个单向关联。
二、Inverse和Cascade的比较
Inverse:负责控制关系,默认为false,也就是关系的两端都能控制,但这样会造成一些问题,更新的时候会因为两端都控制关系,于是重复更新。一般来说有一端要设为true。
Cascade:负责控制关联对象的级联操作,包括更新、删除等,也就是说对一个对象进行更新、删除时,其它对象也受影响,比如我删除一个对象,那么跟它是多对一关系的对象也全部被删除。
举例说明区别:删除“一”那一端一个对象O的时候,如果“多”的那一端的Inverse设为true,则把“多”的那一端所有与O相关联的对象外键清空;如果“多”的那一端的Cascade设为Delete,则把“多”的那一端所有与O相关联的对象全部删除。
发表评论
-
hibernate抓取策略fetch=select /join/subselect
2016-04-10 11:24 584出处:http://blog.csdn.net/ychato ... -
hibernate缓存机制详细分析(一级、二级、查询缓存,非常清晰明白)
2016-04-09 22:23 512收藏自:http://www.360doc.com/cont ... -
hibernate实现JTA事物--代码
2014-11-08 16:15 477package com.ajita.jta; impo ... -
Hibernate的三种连接池设置C3P0、Proxool和DBCP
2014-11-08 16:16 447Xml代码 <!-- JDBC驱动程序 --& ... -
EHCache的使用
2014-05-22 11:39 492在开发高并发量,高性 ... -
缓存 hibernate
2014-06-17 09:40 4601. Session---单数据加载---load/ ge ... -
spring+ehcache实现的缓存查询
2014-06-17 09:40 571最近项目有一个需求,就是用户在查询界面,输入很多查询条件之后, ... -
hibernate + ehcache的例子
2014-05-12 11:23 508这是个hibernate + ehcache的例子,目前使用最 ... -
Hibernate使用EHCache二级缓存 .
2014-05-12 10:30 459数据库结构: create table teamEH ... -
hibernate ehcache
2014-05-12 10:01 4241.EhCache是什么 EhCac ... -
Hibernate中cascade和inverse的作用
2014-05-07 10:38 543Inverse和cascade是Hibernate映射中最难掌 ... -
关联关系
2014-04-22 21:52 336一对一单向外键关联 (学生卡表里有“studentId”字段) ... -
Hibernate一对多,多对一,多对多,一对一关系汇总
2014-05-07 10:42 597一对多 ◆name:集合属性的名称(也可以理解为一对多中那个 ... -
HibernateTemplate+HibernateDaoSupport+SessionFactory
2014-04-19 20:13 507HibernateTemplate @Component ... -
数据库事物
2014-06-17 09:40 3561. 脏读 :脏读就是指当 ... -
Open Session In View
2014-06-17 09:40 452从昨天下午一直纠结到现在,原来是项目启用了Open Sessi ... -
Hibernate主键生成策略
2014-06-20 09:33 2811、自动增长identity 适用于MySQL、DB2、MS ...
相关推荐
- **Configuration**: 配置类的作用及使用方法。 - **SessionFactory**: 创建和管理Session对象。 - **Session**: 进行数据库操作的基本单元。 - **Hibernate高级特性**: - **XDoclet与Hibernate映射**: 自动...
`hibernate-mapping`节点是Hibernate映射文件中的顶级节点,用于定义一系列配置选项,控制整个映射文件的行为和映射规则。这些配置包括数据库连接细节、默认的映射策略等。 - **schema**: 定义了数据库的Schema名称...
在深入探讨Hibernate集合映射中的`inverse`与`cascade`属性之前,我们首先需要理解Hibernate框架的基本概念。Hibernate是一个开放源代码的对象关系映射(ORM)框架,它为Java应用程序提供了一种将对象模型与数据库...
为了使用Hibernate,你需要下载Ant,并将其bin目录添加到系统的PATH环境变量中。 - 下载地址: [Apache Ant](https://ant.apache.org/) 2. **下载Hibernate**: Hibernate提供了多种组件和扩展,确保下载最新的...
- **主要内容**:涉及Hibernate框架的使用方法、实践经验以及作者在Hibernate官方论坛上的交流心得。 - **定位区别**:不同于Hibernate官方参考文档(Hibernate Reference),本文档旨在为开发者提供一个学习和掌握...
### Hibernate Inverse 和 Cascade 的详细讲解 #### 一、引言 在ORM(Object-Relational Mapping)领域,Hibernate作为一款流行的Java持久层框架,它提供了丰富的API和配置选项来帮助开发者实现对象与数据库表之间...
- **工具类**:提供了方便的方法来自动生成数据库表结构,例如使用`Hbm2Ddl`。 #### ID主键生成策略 - **XML方式** - `<generator>`元素用于指定主键生成策略。 - **Annotion方式** - `@GeneratedValue`注解用于...
Hibernate作为Java领域中的一个强大的对象关系映射框架,提供了许多优化数据库操作的策略,其中之一便是延迟加载(Lazy Loading)。延迟加载机制旨在减少不必要的性能消耗,只在真正需要数据时才执行加载操作。本文...
在探讨Hibernate框架中的级联操作(cascade)与控制权反转(inverse)之前,我们需要先对Hibernate有一个基本的理解。Hibernate是一个开放源代码的对象关系映射(ORM)框架,它为Java应用提供了一种将对象模型映射到...
Java面试中关于Hibernate、iBatis和Struts的知识点是评估开发者技能的关键部分。以下是这些框架的核心概念和常见问题的详细解释: **Hibernate** 是一个流行的Java持久层框架,用于简化数据库操作。它提供了对象...
在Java的持久化框架Hibernate中,级联操作(Cascade)和反转(Inverse)是两个重要的概念,它们主要用于管理对象关系模型中的关联关系。在一对多的关系中,这些特性可以帮助简化数据操作,提高代码的可读性和维护性...
在进行Hibernate项目的开发过程中,开发者经常会遇到对Hibernate描述符文件中各种注解(Tag)的应用需求。本文档旨在提供一个全面的@hibernate Tag参考指南,帮助开发人员更好地理解和使用这些标签。 #### 一、@...
在使用了 Hibernate 的系统中,要想在删除某个客户数据的同时删除该客户对应的所有订单数据,可以配置客户和订单关联的 cascade 属性为 all。 7. 一对多关联配置 在一对多关联配置中,存在错误的是 inverse 属性不...
在 Hibernate 中,级联操作(Cascade)和反向属性(Inverse)是管理对象关系的重要概念,特别是在处理一对多(One-to-Many)或多对一(Many-to-One)关系时。 **级联操作(Cascade)** 级联操作定义了当主对象被...
以上便是Hibernate配置文件和类映射文件的基本用法和示例。理解并熟练掌握这些知识,对于构建高效、稳定的Java持久化层至关重要。在实际开发中,开发者可以根据项目需求灵活调整配置,以实现最佳性能和数据管理。
在本篇讨论中,我们将深入探讨如何在Hibernate中配置和使用多对多关系映射,以及其背后的数据库表结构和查询方法。 一、多对多关系的概念与特点 在数据库设计中,多对多关系是指一个表中的记录可以与另一个表中的多...
- Inverse和Cascade: 关联关系中的级联操作。 - 延迟加载(LazyLoading): 减少内存占用和提高性能的技术。 - **事务管理**: - **基于JDBC的事务管理**: - 直接使用JDBC进行事务控制。 - **基于JTA的事务管理**: ...
在Hibernate的配置文件(hibernate.cfg.xml)中,我们需要指定使用的数据库类型(这里是MySQL)及连接参数: ```xml <property name="hibernate.dialect">org.hibernate.dialect.MySQL5Dialect <property name="...