- 浏览: 914783 次
- 性别:
- 来自: 北京
文章分类
- 全部博客 (537)
- Java SE (114)
- Struts (18)
- Hibernate (25)
- Spring (3)
- Page_Tech (41)
- Others (87)
- Database (29)
- Server (24)
- OpenSource_Tools (15)
- IDE_Tool (22)
- Algorithm (28)
- Interview (22)
- Test (28)
- Hardware (1)
- Mainframe (25)
- Web application (4)
- Linux (3)
- PHP (17)
- Android (1)
- Perl (6)
- ubuntu (1)
- Java EE (9)
- Web Analysis (5)
- Node.js (2)
- javascript (2)
最新评论
-
一键注册:
request.getRequestURL()和request.getRequestURI() -
SuperCustomer:
...
SED的暂存空间和模式空间 -
juyo_ch:
讲得挺好理解的,学习了
java 死锁及解决 -
chinaalex:
最后一题答案正确,但是分析有误.按照如下过程,上一行为瓶,下一 ...
zz智力题 -
liaowuxukong:
多谢博主啦,弱弱的了解了一点。
C++/Java 实现多态的方法(C++)
持久层的解决方案有许多,尤其以持久层框架的出现为持久层的开发带来福音。本章以Hibernate和Spring两个最流行的框架来讨论持久层问题,包括Hibernate实体状态、Hibernate关联关系、Hibernate连接表、Spring与JDBC、Spring与Hibernate、Spring与IBatis.
6.1 问题:Hibernate中的实体状态
6.1.1 怎样理解实体状态
程序员M在使用Hibernate时总是有些惊奇,原因在于Hinberate中对各个实体的状态有着不可思议的控制魔力。比如:有时候在更改实体的属性时,与数据库中对应的列值也会被更改,而有时候又无法对应更新起来。想要更新的时候无法更新,不想更新的却更新了。
这就引发了程序员M的问题:在什么情况下这个实体是持久化一致的,什么时候又不一致呢?在Hibernate中实体有三个状态:瞬时、持久化和脱管。要想解决持久化一致性的问题,就必须了解各个状态各代表了什么,而且在什么操作后这些状态会发生变化。
6.1.2 实体状态的定义
Hibernate中实体有三个状态:瞬时、持久化和脱管。下面先来看看Hibernate对这三个状态是怎么定义的。
(1)瞬时:一个实体通过new操作符创建后,没有和Hibernate的Session建立关系,也没有手动赋值过该实体的持久化标识(持久化标识可以认为映射表的主键)。此时该实体中的任何属性的更新都不会反映到数据库表中。
(2)持久化:当一个实体和Hibernate的Session创建了关系,并获取了持久化标识,而且在Hibernate的Session生命周期内存在。此时针对该实体任何属性的更改都会直接影响到数据库表中一条记录对应字段的更新,也即与对应数据库表保持同步。
(3)脱管:当一个实体和Hibernate的Session创建了关系,并获取了持久化标识,而此时Hibernate的Session的生命周期结束,实体的持久化标识没有被改动过。针对该实体的任何属性的修改都不会及时反映到数据库表中。
这三种状态有两个很重要的点需要掌握:Hibernate的Session和持久化标识。通过这两个条件就可以判断出究竟是3种状态中的哪一个。3种不同状态通过Hibernate的Session和持久化标识可以互相之间进行转换,如图6.1所示。
图6.1 3个状态的转化
举个简单的例子,假如Room实体的属性id表示持久化标识,那么:
(1)创建的Room实例为瞬时状态,将表中对应的主键手动设置到id属性,此时就是脱管状态。
(2)创建的Room实例为瞬时状态,不设置id或设置的id在表中找不到对应,此时调用Hibernate Session的持久化方法,将成为持久化状态。
(3)Room实体在持久化状态下关闭Hibernate Session,此时就是脱管状态。
(4)Room实体在脱管状态下调用Hibernate Session的持久化方法,此时就是持久化状态。
(5)Room实体在持久化状态下关闭Hibernate Session,随后清除id属性的值,此时就是瞬时状态。
6.1.3 实体状态的代码实现:瞬时—持久化
从瞬时状态到持久化状态,Hibernate提供了如下的实现,见例6.1(以下代码省略了配置映射文件的部分)。
例6.1:瞬时—持久化的实现
public void run() {
//创建瞬时状态的UserInf实例
UserInfo userInfo = new UserInfo();
//设置UserInfo属性,持久化标识id属性在映射中为自增长,不用设置
userInfo.setName("RW");
userInfo.setSex("M");
//启动Session
Session session = HibernateSessionFactory.currentSession();
//启动事务
Transaction tx = session.beginTransaction();
//瞬时—持久化的实现,保存UserInfo代表的一条记录到数据库
①session.save(userInfo);
//打印结果
System.out.println("---Id:" + userInfo.getId());
System.out.println("---Name:" + userInfo.getName());
System.out.println("---Sex:" + userInfo.getSex());
//对持久化的UserInfo进行属性的更新,此时将同步数据库
②userInfo.setName("RW2");
userInfo.setSex("F");
//不用调用update方法,持久化状态的UserInfo会自动同步数据库
//打印结果
System.out.println("---Id:" + userInfo.getId());
System.out.println("---Name:" + userInfo.getName());
System.out.println("---Sex:" + userInfo.getSex());
//提交事务
tx.commit();
//关闭Hibernate Session
HibernateSessionFactory.closeSession();
}
针对该段代码将执行如下SQL语句:
Hibernate:
/* ①session.save(userInfo);的动作 */
insert
into
userinfo
(NAME, SEX, roomid, id)
values
(?, ?, ?, ?)
Hibernate:
/* ②userInfo.setName("RW2");
userInfo.setSex("F"); 的动作*/
update
userinfo
set
NAME=?,
SEX=?,
roomid=?
where
id=?
当瞬时状态转变为持久化状态时,需要自行调用持久化方法(如:session.save())来执行SQL。而在持久化状态时,Hibernate控制器会自动侦测到改动,执行SQL同步数据库。
6.1.4 实体状态的代码实现:脱管-持久化、持久化-脱管
从脱管状态和持久化状态双重转变,Hibernate提供了如下的实现,见例6.2(以下代码省略了配置映射文件的部分)。
例6.2:脱管状态和持久化状态双重转变
public void run() {
//创建UserInfo实例
UserInfo userInfo = new UserInfo();
//启动Session
Session session = HibernateSessionFactory.currentSession();
//启动事务
Transaction tx = session.beginTransaction();
//得到持久化UserInfo,此时UserInfo为持久化状态
//与数据库中主键为11117的记录同步
①session.load(userInfo,new Long(11117));
//提交事务
tx.commit();
//关闭Hibernate Session
HibernateSessionFactory.closeSession();
//关闭Hibernate Session后UserInfo的状态为脱管状态
//此时依然能够得到数据库在持久化状态时的数据
System.out.println("---Id:" + userInfo.getId());
System.out.println("---Name:" + userInfo.getName());
System.out.println("---Sex:" + userInfo.getSex());
//对userInfo实体的属性的操作将不影响数据库中主键为11117的记录
②userInfo.setName("RW3");
userInfo.setSex("M");
//启动Session
session = HibernateSessionFactory.currentSession();
//启动事务
tx = session.beginTransaction();
//从脱管状态到持久化状态的转变,此时将更新数据库中对应主键为11117的记录
③session.update(userInfo);
//提交事务
tx.commit();
//关闭Hibernate Session
HibernateSessionFactory.closeSession();
}
针对该段代码将执行如下SQL语句:
Hibernate:
/* ①session.load(userInfo,new Long(11117))的动作 */
select
userinfo0_.id as id0_0_,
userinfo0_.NAME as NAME0_0_,
userinfo0_.SEX as SEX0_0_,
userinfo0_.roomid as roomid0_0_
from
userinfo userinfo0_
where
userinfo0_.id=?
Hibernate:
/* ③session.update(userInfo)的动作 */
update
userinfo
set
NAME=?,
SEX=?,
roomid=?
where
id=?
可以看到②userInfo.setName("RW3")这一部分的代码没有直接同步数据库的表,因为此时Hibernate Session已经关闭了,此时是脱管状态。而直到再次打开Hibernate Session并调用③session.update(userInfo),此时由于持久化标识存在于UserInfo实例,因此将从脱管状态转变为持久化状态,同步数据库。
6.1.5 持久化方法对状态的影响
在Hibernate中定义了多个持久化方法,这些方法的调用对实体状态是有影响的。注意,并不是每一个持久化方法都会将实体状态变为持久化状态。在之前的代码中,已经使用到的持久化方法为session.save()、session.load()、session.update()。下面是另外一些持久化方法的调用方式。
(1)session.delete()方法
该方法将已经存在的表记录删除,其所影响的状态是从持久化、脱管状态转变为瞬时状态,见例6.3。
例6.3:session.delete()方法对状态的变化
public void run() {
//创建UserInfo实例
UserInfo userInfo = new UserInfo();
//启动Session
Session session = HibernateSessionFactory.currentSession();
//启动事务
Transaction tx = session.beginTransaction();
//得到持久化UserInfo,此时UserInfo为持久化状态
//与数据库中主键为11117的记录同步
①session.load(userInfo,new Long(11117));
//删除持久化状态的UserInfo实体,此时UserInfo实体为瞬时状态
②session.delete(userInfo);
//提交事务
tx.commit();
//关闭Hibernate Session
HibernateSessionFactory.closeSession();
//由于执行了session.delete因此UserInfo实体为瞬时状态,在数据库中找不到主键为11117的数据
//此时依然能够显示该实体的属性
System.out.println("---Id:" + userInfo.getId());
System.out.println("---Name:" + userInfo.getName());
System.out.println("---Sex:" + userInfo.getSex());
//更新UserInfo实体的持久化标识,使其成为脱管状态
③userInfo.setId(11116);
//启动Session
session = HibernateSessionFactory.currentSession();
//启动事务
tx = session.beginTransaction();
//调用delete方法将脱管状态的UserInfo实体转变为瞬时状态
④session.delete(userInfo);
//提交事务
tx.commit();
//关闭Hibernate Session
HibernateSessionFactory.closeSession();
}
针对该段代码将执行如下SQL语句:
Hibernate:
/* ①session.load(userInfo,new Long(11117))的动作 */
select
userinfo0_.id as id0_0_,
userinfo0_.NAME as NAME0_0_,
userinfo0_.SEX as SEX0_0_,
userinfo0_.roomid as roomid0_0_
from
userinfo userinfo0_
where
userinfo0_.id=?
Hibernate:
/* ②session.delete(userInfo)的动作 */
delete
from
userinfo
where
id=?
Hibernate:
/* ④session.delete(userInfo)的动作 */
delete
from
userinfo
where
id=?
可以看到,两句delete语句分别对应了持久化状态的UserInfo和脱管状态的UserInfo的删除动作。之后两种状态的UserInfo都会成为瞬时状态。
发表评论
-
hibernate n+1问题
2010-10-21 11:05 931Hibernate中常会用到set,bag等集合表示1对多的关 ... -
Hibernate 关联关系 总结
2010-09-09 15:27 9581.一对多的单向关联关系 配置单向的一对多关系是 ... -
Hibernate 关联
2010-09-09 15:24 10011、hibernate多对一关联映 ... -
Hibernate一对多(单向)
2010-09-09 14:31 594[正文]: Hibernate一对多关联,例如一个用户有 ... -
Hibernate中No row with the given identifier exists问题的原因及解决
2010-06-23 09:54 935产生此问题的原因: ... -
Hibernate使用count(*)取得表中记录总数(跨Hibernate3.x版本问题)
2010-06-22 17:13 1530Java代码 /** * @T ... -
hibernate继承关系映射
2010-06-13 16:58 919hbn 的继承映射关系有这 ... -
Hibernate集合映射
2010-06-13 12:49 912准备找工作,重新整理一下Hibernate,今天做了集合映射 ... -
高并发网站的架构
2010-05-07 11:07 715我在CERNET做过拨号接入平台的搭建,而后在Yaho ... -
Hibernate事务和并发控制
2010-05-07 10:21 9141. 事务介绍:1.1. 事务的定义:事务就 ... -
hibernate中lazy的使用
2009-12-18 22:00 779lazy,延迟加载 Lazy的 ... -
Hibernate中代码自动生成功能小结
2009-12-06 15:10 1035Hibernate中需要class和mapping file, ... -
hibernate工具箱—根据映射文件自动建表
2009-12-04 12:08 976public class ExportDB { ... -
关联加载对象时的报错-----a different object with the same identifier value
2009-11-18 16:13 884因为在hibernate中同一个session里面有了两个相同 ... -
update/saveOrUpdate/merge
2009-11-18 15:28 1140通常下面的场景会使用update()或saveOrUpdate ... -
写得很不错的-Hibernate中的实体状态(二)
2009-11-18 15:08 949(2)session.merge ()方法 ... -
Hibernate3.x总结
2009-11-18 14:29 751Hibernate不是盏省油的灯 ... -
Hibernate Annotation几种关联映射
2009-11-02 17:01 1076Hibernate Annotation几种关联映射 一对一 ... -
hibernate3的注解映射学习
2009-11-02 16:41 1345注解映射必须满足两大条件:Hibernate3.2以上版本和J ... -
Hibernate 中级联操作 cascade 选项
2009-11-02 16:35 926none :在保存、更新或删除对象时,忽略其他关联的对象。他是 ...
相关推荐
本文将深入探讨Hibernate中的实体状态及其转换,帮助开发者更好地掌握如何有效地使用Hibernate进行数据操作。 首先,我们要了解Hibernate定义的三种实体状态: 1. **临时状态(Transient)**:当我们在程序中通过`...
4. **持久化上下文(Persistence Context)**:是JPA管理实体实例的环境,它可以跟踪实体的状态(新、已修改、已加载或已删除)并负责与数据库的交互。 5. **查询语言**:JPA提供JPQL(Java Persistence Query ...
- 实体类和注解:创建实体类并使用Hibernate注解进行字段映射。 - Session工厂与Session:理解Session工厂的概念,以及Session在数据操作中的角色。 2. **Day02:持久化操作** - CRUD操作:掌握如何通过...
- **查询特定修订时刻的实体状态**:介绍了如何查询某一特定修订时刻实体的状态。这对于回溯系统中的某个时间点非常有用。 - **查询实体变更修订**:展示了如何查询实体发生变更的具体修订时刻。这对于了解实体何时...
2. **持久化层**:Hibernate 为 Java 对象提供了一种持久化的机制,使得对象的状态能够在数据库中保存和恢复。 3. **查询语言 (HQL)**:Hibernate 查询语言允许开发者用面向对象的方式编写查询,而不是直接写 SQL。...
总的来说,Hibernate实体层设计是一个综合性的任务,它涉及到对象模型、数据库映射、关系处理、性能优化、事务管理等多个方面。理解和熟练掌握这些知识,能够帮助我们构建出高效、稳定的Java企业应用。
5. 关联映射:Hibernate支持一对一、一对多、多对一和多对多的关联映射,使得处理对象间的关联变得简单。 综上所述,Java培训-Hibernate课程旨在让开发者熟练掌握使用Hibernate进行数据库操作的技巧,通过理解并...
10. **实体状态管理**:持久化对象的瞬时态、持久态、游离态和脱管态的解释及其转换。 11. **JPA与Hibernate的关系**:简述Java Persistence API (JPA)与Hibernate之间的联系和差异。 这些知识点覆盖了Hibernate的...
10. **实体状态管理**:解析瞬时态、持久态、托管态和脱管态四种实体状态,以及它们之间的转换。 以上是对“Hibernate3学习笔记(一)-hibernate概述和简单实例入门”可能涵盖的详细知识点的解读,每个主题都可以深入...
在Hibernate中,`inverse`属性是一个关键的概念,它用于定义一对多或多对一关系中的维护端。`inverse="true"`表示另一端负责维护关联关系,这在处理关联实体的保存和更新时具有重要意义。例如,如果你有一个学生类和...
Hibernate 是一个开源的对象关系映射(ORM)框架,它允许Java开发者将数据库操作与业务对象进行解耦,简化了数据库编程。`hibernate-core`是Hibernate的核心库,包含了执行ORM功能所需的主要组件。 在Hibernate-...
通过创建一个简单的Java项目,引入Hibernate库,并编写一个实体类与数据库表对应,你会了解如何初始化SessionFactory,创建Session,以及执行基本的保存和查询操作。 2. **Hibernate原理模拟 - O/R Mapping** O/R ...
《Hibernate in Action》是Manning出版社出版的一本深入解析Hibernate框架的专业书籍,它全面地介绍了如何在Java应用程序中高效地使用Hibernate进行数据持久化。Hibernate是一个开源的对象关系映射(ORM)框架,它极...
《夏昕-Hibernate 开发指南》是一本专为程序员设计的深入学习Hibernate框架的书籍。Hibernate作为Java领域中广泛使用的对象关系映射(ORM)工具,它极大地简化了数据库操作,使得开发者可以更加专注于业务逻辑而...
8. **实体状态管理**:Hibernate3将对象分为瞬时态、持久态、托管态和脱管态四种状态,理解这些状态有助于更好地掌握对象的生命周期。 9. **性能优化**:包括延迟加载(Lazy Loading)、批处理(Batch Processing)...
14. 理解JavaBean对象在Hibernate应用中的不同状态。 15. 熟悉处理关联关系的域模型对象持久化前的准备工作。 16. 理解映射文件中cascade属性,了解不同取值的含义和用法。 17. 明白lazy属性的含义,掌握其在优化...
在压缩包中,"hibernate-orm-master"文件夹很可能是包含了整个Hibernate ORM的源代码仓库,这对于开发者来说是一个宝贵的学习资源。你可以通过阅读源码来理解Hibernate的内部工作机制,比如实体管理、会话管理、缓存...
在Java的持久化框架Hibernate中,Many-to-Many关系是一种常见的数据库表之间的关联方式,它表示一个实体可以与多个其他实体进行关联,反之亦然。本文将深入探讨如何在Hibernate中处理Many-to-Many关系的级联保存、...