我的印象里, Hibernate session中常用的保存操作只有:save, update, saveOrUpdate,delete;还有其他很多操作没有用过:persist(), merge(), lock(), refresh(), evict(), replicate() ,有必要弄清楚这些方法分别是做什么, 实体对象的状态会发生什么变化?
先看看这些方法中,从javadoc的注释可以得出:
临时 —> 持久
save,persist,saveOrUpdate
托管 —> 持久
update,saveOrUpdate,saveOrUpdateCopy , merge,lock,replicate,refresh
持久 —> 托管
evict
持久 —> 临时
delete
根据以上,可以看出托管->持久的方法是最多的。
那么以上各类情况下,各个方法有什么区别?应该怎样取舍?在hibernate3.2环境下,我得出以下结论:
1. save和perisit
几乎一样,都是持久化一个临时对象;
只是persist持久化的时间可能推迟到flunsh之前,而且perisit传入的对象为托管对象时,会报错,而save托管对象不会报错只是将id置null。
可以看出,在持久化一个临时对象时,只用save就行了。
2. saveOrUpdate和merge
saveOrUpdate将unsaved对象持久化或托管对象持久化;
merge,从名字也能猜到是合并,跟saveOrUpdate相似,可以持久化托管和临时对象。
不同的是:在更新时候,merge将托管对象的属性复制到session中有相同标识符(但不相等)的持久对象(如果不存在,就从db load),也就是先select 后 update;倘若找不到相同标识符的对象,则置id为null,保存临时对象。返回的是持久对象,但是merge传入的对象还是托管的。
可以看出,只有在session中已经存在一个具有相同标识符的持久对象的时候,应该采用merge,此时用saveOrUpdate会报错。当然如果碰到这类情形,将对象属性copy到已经持久化的那个相同标识符对象上也是可以的,正因为如此我一般都没用过merge。再者merge在session中不存在相同标识对象时,会多一条select,往往我们并不需要。
3. merge和replicate
replicate,通过复制来持久化当前已经脱离session的托管对象,这点上看有点像merge,但是当传入一个临时对象则会报错,而且replicate后的传入的那个对象将是持久的,这也是跟merge不同。
通过参数ReplicationMode可以控制遇到相同记录时的行为。
如果是 ReplicationMode.EXCEPTION,则在复制时如果有重复的行数据,则抛出异常。ReplicationMode.IGNORE则忽略 异常。
ReplicationMode.OVERWRITE则会覆盖掉已有行数据。ReplicationMode.LATEST_VERSION则是在有重复时使用最新的版本进行控制。
可以看出,replicate在执行复制的时候,比起merge还是有更多可控行为,ReplicationMode.LATEST_VERSION还可以保证并发修改时只更新最新的版本。
4. lock和refresh
相类似的,都将托管对象又重新持久化,而且可以指定查询时候的锁:
* 当用户显式的使用数据库支持的SQL格式SELECT ... FOR UPDATE 发送SQL的时候,锁定级别设置为LockMode.UPGRADE
* 当用户显式的使用Oracle数据库的SQL语句SELECT ... FOR UPDATE NOWAIT 的时候,锁定级别设置LockMode.UPGRADE_NOWAIT
*当Hibernate在“可重复读”或者是“序列化”数据库隔离级别下读取数据的时候,锁定模式 自动设置为LockMode.READ。这种模式也可以通过用户显式指定进行设置。
*LockMode.NONE 代表无需锁定。在Transaction结束时, 所有的对象都切换到该模式上来。与session相关联的对象通过调用update() 或者saveOrUpdate()脱离该模式。
以上只有 LockMode.NONE,先从缓存查找持久对象;其他的都直接从数据库查找。
有些不同的是,lock还有检查对象版本的能力,如果指定的锁定模式是READ, UPGRADE 或 UPGRADE_NOWAIT,那么Session.lock()就 执行版本号检查。
refresh提供了比load更方便的查找能力,一个刚脱离了session的持久对象,如果用load还需要用id查找,那么refresh就可以直接传入托管对象刷新其状态了。
5. delete和evict
evict,从session的缓存中去除当前实例。执行后对象的改变将不再和数据库保持同步。当指定级联风格为'evict’时,会级联操作关联对象。在用于批量操作的时候,清空缓存,防止内存紧张。
delete,也会从session的缓存中去除当前实例,但flunsh时会执行数据库delete,之后对象就成了临时状态。
可以看出delete比起evict,不仅从session删除,还会从数据库删除。
参考文献:
http://www.redsaga.com/hibernate-ref/3.x/zh-cn/html/objectstate.html
http://loveexception.iteye.com/blog/29353
分享到:
相关推荐
在Java的ORM框架Hibernate中,实体管理是其...理解并熟练运用Hibernate的实体状态转换,对于优化数据操作和提高代码效率至关重要。在实际开发中,开发者应根据业务需求选择合适的转换方式,确保数据的正确性和一致性。
- **保存和插入**:`save()`方法用于将瞬时对象转换为持久化对象,而`persist()`方法在JPA中具有相同的功能,但在某些情况下,如已存在ID,它可能不会立即触发SQL INSERT语句。 - **更新**:调用`update()`方法将...
对象通过调用Session的save()或persist()方法后,会从临时状态变为持久化状态。这时,Hibernate会给对象分配一个唯一的ID(通常是自增ID),并将对象的信息保存到缓存中。当Session执行flush操作时,会将缓存中的...
除了状态转换,Hibernate 还提供了一些方法来处理对象的保存: - **save()**:用于保存新对象,如果对象已经是持久态或脱管态,那么它将更新数据库中的记录。 - **persist()**:与 `save()` 类似,但不立即触发 SQL...
SSHnote session函数的对象状态转换是Hibernate框架中一个关键的概念,涉及到如何管理数据库中的持久化对象。在Hibernate中,对象有三种基本状态:瞬时(Transient)、持久化(Persistent)和脱管(Detached)。理解...
本篇文章将深入探讨如何实现自己的Hibernate框架,特别是Session方法的实现,这对于理解ORM的工作原理及优化数据库访问逻辑具有重要意义。 首先,我们需要了解Hibernate的核心概念——Session。在Hibernate中,...
另外,`Session`还有一系列方法用于处理对象的状态转换,如`lock()`、`merge()`、`refresh()`、`evict()`等,这些方法在处理对象生命周期的不同阶段时非常关键。 `Configuration`和`SessionFactory`也是Hibernate的...
- 可以通过调用 `Session` 的 `save()` 或 `saveOrUpdate()` 方法将瞬时对象转换为持久对象,此时对象的状态会与数据库同步。 2. **持久对象(Persistent Objects)** - 持久对象拥有唯一的数据库标识(ID),...
3. **对象状态转换**:在`flush()`操作中,Hibernate可能将瞬态对象转换为持久化对象,或将持久化对象的状态更新到数据库。 4. **批量操作优化**:为了提高效率,`flush()`可能会合并多个小的更新或插入操作,形成...
**自由态**是对象刚刚被创建,还没有被Hibernate管理,或者从Session中清除后返回的状态。此时,对象与数据库没有关联,其状态不会自动同步到数据库。如果需要将自由态对象保存到数据库,必须调用Session的save()或...
理解并掌握Hibernate的这三种状态转换对于开发者来说非常重要,因为它直接影响到对象的生命周期管理以及如何有效地更新数据库中的数据。在实际开发中,正确处理这些状态转换可以避免数据一致性问题,提高代码的可...
理解这些状态及其转换对于掌握 `Session` 的工作原理至关重要。 总之,`Session` 在Hibernate框架中的作用非常重要,不仅是因为它可以高效地管理数据访问,还因为它通过缓存机制极大地提高了应用程序的性能。理解和...
其次,持久状态(Persistent)意味着对象已经通过Session的save()、saveOrUpdate()等方法保存到数据库,并且仍然在Hibernate的缓存管理中。在这种状态下,对象的任何改动在事务提交时会被自动同步到数据库。例如,...
- **持久化状态 → 游离状态**:当关闭 Session 或调用 `evict()` 方法时,持久化对象会变为游离状态。 - **游离状态 → 持久化状态**:可以通过再次调用 `save()`、`persist()` 或 `merge()` 方法将游离对象转变为...
4. 状态转换: 瞬时态到持久态的转换通常通过`save()`或`persist()`完成。持久态对象在Session关闭后会自动变为脱管态。脱管态对象可以通过`merge()`或`saveOrUpdate()`恢复为持久态。值得注意的是,`saveOrUpdate()...
在 Hibernate 框架中,Java 对象的状态管理和 Session 缓存是核心概念,它们直接影响着数据的持久化过程和数据库...同时,根据具体应用需求,选择合适的缓存清理策略和对象状态转换时机,以达到最佳的数据库操作效果。
通过Session的save()或persist()方法,我们可以将瞬时对象转换为持久对象,确保其状态被持久化到数据库。 **持久化对象的生命周期:** - **瞬时态(Transient)**:对象刚刚创建,尚未与Session关联,此时对象状态...
Hibernate允许我们将对象状态转换为XML格式,这在多种场景下非常有用,例如在需要将数据导出到外部系统进行处理时,或者在需要支持数据的多种格式时。 在Hibernate中使用XML数据存储,通常涉及到以下几个关键概念和...
二、状态转换与操作 - **瞬时态 -> 持久态**:通过Session的save()、saveOrUpdate()或persist()方法,瞬时态对象会获得一个数据库生成的ID,变为持久态。 - **持久态 -> 瞬时态**:当Session关闭或调用clear()...
4. **使用Hibernate session**:在操作数据时,需先调用`open()`打开Session,完成后及时调用`close()`关闭Session,以释放资源并让连接池回收连接。 5. **Integer与int映射的差异**:使用Integer作为映射类型,...