`
topquan
  • 浏览: 30100 次
  • 性别: Icon_minigender_1
  • 来自: 广州
最近访客 更多访客>>
社区版块
存档分类
最新评论

hibernate数据加载

阅读更多
单一数据加载:Session.get/load
均可以根据指定的实体类和id从数据库中读取记录,并返回与之对应的实体对象。
区别:
1.如果未能发现符合条件的记录,get方法返回null,而load方法会抛出一个ObjectNotFoundException。
2.Load方法可返回实体的代理类实例,而get方法永远直接返回实体类。
3.load方法可以充分利用内部缓存和二级缓存中的现有数据,而get方法仅在内部缓存中查找,如果没有发现对应的数据,将越过二级缓存,直接调用SQL完成数据读取。
数据加载的过程:
1.在一级缓存中,根据实体类型和id进行查找,如果在第一级缓存中命中,且数据状态合法,则直接返回。
2.Session会在当前"NonExists"记录中进行查找,如果在"NonExists"记录中存在同样的条件,返回null。
3.对load方法而言,如果内部缓存中没法现有效数据,查询二级缓存,命中则返回。
4.如果缓存中无有效数据,发起数据库查询操作(Select SQL),如果经过查询未发现对应记录,将此次查询的信息在"NonExists"中加以记录,返回null.
5.根据映射配置和Select SQL查询得到的ResultSet,创建对应的数据对象。
6.将对象纳入一级缓存。
7.执行Interceptor.onLoad方法(如果有对应的Interceptor)
8.将数据纳入二级缓存
9.如果数据对象实现了LifeCycle接口,则调用数据对象的onLoad方法。
10.返回数据对象。


批量查询:Session.find/iterate
查询性能往往是系统性能表现的一个重要方面,查询机制的优劣很大程度上决定了系统的整体性能。这个领域往往也存在最大的性能调整空间。

hibernate2中Session.find()对应于3中的session.createQuery().list();
hibernate2中Session.iterate()对应于3中的session.createQuery().iterate();
find和iterate区别:
find方法通过一条Select SQL实现了查询操作,而iterate方法要执行多条Select SQL.
iterate第一次查询获取所有符合条件的记录的id,然后再根据各个id从库表中读取对应的记录,这是一个典型的N+1次的查询问题,如果符合条件记录有10000条,就需要执行10001条Select SQL,可想性能会如何的差。

那为什么要提供iterator方法,而不只是提供高效率的find方法?

原因1.与hibernate缓存机制密切相关
find方法实际上是无法利用缓存的,它对缓存只写不读。
find方法只执行一次SQL查询,它无法判断缓存中什么样的数据是符合条件的,也无法保证查询结果的完整性。而iterate方法,会首先查询所有符合条件记录的id,然后根据id去缓存中找,如果缓存中有该id,就返回,没有可以根据id再去数据库查询。
String hql = "from TUser where age > ?";
List userList = session.find(hql, new Integer(18), Hibernate.INTEGER);
Iterator it = session.iterate(hql, new Integer(18), Hibernate.INTEGER);
顺序执行,iterate方法只会执行一次SQL查询,就是查找id,然后根据id就可以从缓存中获得数据。

String hql = "from TUser where age > ?";
List userList = session.find(hql, new Integer(18), Hibernate.INTEGER);
userList = session.find(hql, new Integer(18), Hibernate.INTEGER);
缓存是不起作用的。
如果目标数据读取相对较为频繁,通过iterate这种机制,会减少性能损耗。

原因2.内存使用上的考虑
find方法将一次获得的所有记录并将其读入内存。如果数据量太大,可能会触发OutOfMemoryError,从而导致系统异常。解决方案之一就是结合iterate方法和evict方法逐条对记录进行处理,将内存消化保持在一个可以接受的范围之内。如:
String hql = "from TUser where age > ?";
Iterator it = session.iterate(hql, new Integer(18), Hibernate.INTEGER);
while(it.hasNext()) {
    TUser user = (TUser)it.next();
   
    //将对象从一级缓存中删除
    session.evict(user);

    //二级缓存可以设定最大缓存量,达到后自动对较老数据进行废除,但也可以通过编
    //码移除,这样有助于保持数据有效性。
    sessionFactory.evict(TUser.class, user.getID());
}


批量数据处理的缓存同步问题
1.hibernate 2:
session.delete("from TUser");
会先查询出id,然后逐个id执行 delete from T_User where id = ?;
这样造成效率低下。
为什么不直接采用一条Delete SQL?是因为ORM要自动维持其内部状态属性,必须知道用户作了什么操作。必须先从数据库中获得待删除对象,然后根据这些对象对内部缓存和二级缓存的数据进行整理,以保持内存状态与数据库的一致性。
单执行一条删除语句,删除了什么数据,只有数据库知道,ORM无法得知。下次用户从缓存中读出的数据,很可能就是被删除的数据,从而导致逻辑错误。当然,如果ORM可以根据DELETE SQL对缓存中数据进行处理,将缓存中符合条件的对象废除,然后再执行DELETE SQL
,但是这样导致缓存的管理复杂性大大增加(实际相当于实现了一个支持SQL的内存数据库),这对于轻量级的ORM实现而言太苛刻了。
2.hibernate 3
性能提高。
但无法解决缓存同步上的问题,无法保证缓存数据的一致有效性。
Tuser user = (TUser)session.load(TUser.class, new Integer(1));

//通过Bulk delete/update 删除id=1的用户记录
Transaction tx = session.beginTransaction();
String hql = "delete TUser where id=1";
Query query = session.createQuery(hql);
query.executeUpdate();
tx.commit();

//再次尝试加载
user = (TUser)session.load(TUser.class, new Integer(1));
可以看到第二次加载是成功的。



topquan 2006-08-05 01:19 发表评论
分享到:
评论

相关推荐

    Hibernate之数据加载方式

    **标题:“Hibernate之数据加载方式”** 在Java的持久化框架中,Hibernate是一个非常重要的工具,它简化了数据库操作,提供了对象关系映射(ORM)的功能。本篇将深入探讨Hibernate的数据加载方式,帮助开发者更好地...

    Hibernate延迟加载以及利用Spring

    ### Hibernate延迟加载以及利用Spring #### 一、Hibernate延迟加载概念与原理 在理解Hibernate的延迟加载机制之前,我们首先需要了解什么是延迟加载。延迟加载(Lazy Loading)是一种设计模式,其核心思想是在真正...

    hibernate 延迟加载深入剖析

    当程序试图访问这些属性时,Hibernate会在后台异步加载实际的数据。 #### 三、延迟加载的具体应用场景 ##### 3.1 集合属性的延迟加载 集合属性的延迟加载是延迟加载中最常见的应用场景之一。当一个实体拥有大量的...

    Hibernate 延迟加载剖析与代理模式应用

    为了避免这种情况,Hibernate提供了延迟加载机制,使得只有在真正需要访问`addresses`时,才会去数据库查询相关的`Address`数据。 配置延迟加载的关键在于Hibernate的映射文件。在`<set>`元素中,我们通常会设置`...

    hibernate延迟加载解决

    ### Hibernate延迟加载详解 #### 一、什么是延迟加载? 延迟加载是一种优化技术,在软件开发中广泛应用于各种场景,尤其在...通过合理配置和使用这一机制,可以有效地优化应用程序的数据加载流程,提高用户体验。

    hibernate懒加载策略.doc

    综上所述,Hibernate的懒加载策略是通过代理对象和回调机制实现的,旨在按需加载数据,减少不必要的数据库操作,提高系统效率。合理使用懒加载,可以显著优化数据访问性能,但同时也需要考虑其可能带来的问题和挑战...

    集成spring的hibernate懒加载

    6. **使用Spring Data的`@EntityGraph`**:允许你在查询时定义数据加载的图形,更精细地控制加载行为。 7. **使用`@Transactional(readOnly = true)`**:对于只读操作,可以使用只读事务,这样即使在Session关闭后...

    Hibernate延时加载与lazy机制.doc

    - **数据加载**:首次访问代理类的非ID属性时,Hibernate会执行相应的SQL查询,从数据库中加载数据,并将代理类实例初始化为实际对象。 **5. 示例分析** 以下代码展示了延迟加载的过程: ```java tx = session....

    Hibernate数据持久层jar包源码解析

    《Hibernate数据持久层jar包源码解析》 在软件开发中,数据持久层是系统架构中的关键部分,它负责将应用程序的数据与数据库进行交互。Hibernate作为一款强大的Java对象关系映射(ORM)框架,极大地简化了这个过程。...

    hibernate延迟加载技术详细解

    - Hibernate 自动为懒加载属性创建代理类,这些代理类在需要时才加载真实数据。 #### 四、总结 在 Hibernate 中,延迟加载技术是非常关键的一部分,它能够显著提高应用程序的性能。通过合理地选择 Fetching 策略...

    Hibernate lazy加载FOR Connection

    标题中的“Hibernate lazy加载FOR Connection”指的是Hibernate框架中的一种特性,即懒加载(Lazy Loading)。在Hibernate中,懒加载是一种优化策略,它推迟对关联对象的加载,直到真正需要使用这些对象时才进行加载...

    hibernate 延迟加载.docx

    2. **内存优化**:避免了不必要的数据加载,节省了内存资源,防止内存溢出。 **立即加载与延迟加载的对比** - **立即加载(Eager Loading)**:使用`Session.get()`方法获取对象时,会同时加载与其关联的对象,...

    Hibernate加载方式与多态加载分析

    本篇文章将探讨Hibernate的加载方式以及多态加载的概念。 首先,我们关注Session中的`load`和`get`方法。`load`方法是用于根据主键加载对象,它支持CGLIB懒加载(lazy loading)动态代理。当lazy属性设置为true时,...

    hibernate抓取策略和懒加载案例

    在Java的持久化框架Hibernate中,数据访问优化是至关重要的,而抓取策略(Fetch Strategy)和懒加载(Lazy Loading)则是实现这一目标的关键技术。本文将深入探讨这两个概念,并通过具体的案例进行分析。 首先,让...

    Hibernate的延迟加载

    这样,Hibernate可以在集合尚未被访问时,避免执行关联的数据加载。 启用集合类型的延迟加载,同样在映射配置文件中进行设置: ```xml <hibernate-mapping> ... </hibernate-mapping> ``` 考虑以下...

    Hibernate 加载数据库驱动的 Jar

    在这个场景中,"Hibernate加载数据库驱动的Jar"是指在 Hibernate 应用程序中添加 SQL Server 驱动的 JAR 文件,以便正确地连接到SQL Server数据库。 首先,我们来看一下给定的三个JAR文件: 1. **msbase.jar**:这...

    【北大青鸟内部教程】jsp中关于Hibernate延时加载的问题

    在处理大数据量或者复杂数据结构时,Hibernate引入了延时加载(Lazy Loading)机制,以提高应用程序的性能。本文将深入探讨在JSP(JavaServer Pages)环境中,如何理解和应对Hibernate的延时加载问题。 首先,我们...

    Hibernate延迟加载介绍.doc

    如果目标对象(即真实的User对象)尚未加载,Hibernate会执行SQL查询(如`select * from user where id='1'`),获取数据并构建目标对象,然后将其赋值给代理对象的`CGLIB$CALLBACK_0.target`属性。 二、集合类型的...

Global site tag (gtag.js) - Google Analytics