Hibernate之lazy延迟加载(转)

一.延迟加载的概念

当Hibernate从数据库中加载某个对象时,不加载关联的对象,而只是生成了代理对象,获取使用session中的load的方法(在没有改变lazy属性为false的情况下)获取到的也是代理对象,所以在上面这几种场景下就是延迟加载。

二.理解立即加载的概念

当Hibernate从数据库中加载某个对象时,加载关联的对象,生成的实际对象,获取使用session中的get的方法获取到的是实际对象。

三.为什么要使用延迟加载

延迟加载策略能避免加载应用程序不需要访问的关联对象,以提高应用程序的性能。

四.立即加载的缺点

Hibernate在查询某个对象时,立即查询与之关联的对象,我们可以看出这种加载策略存在两大不足:

1.select的语句数目太多,需要频繁的访问数据库,会影响查询的性能。

2.在应用程序只需要访问要的对象,而不需要访问与他关联的对象的场景下,加载与之关联的对象完全是多余的操作,这些多余的操作是会占内存,这就造成了内存空间的浪费。

五.什么时候使用延迟加载什么时候使用立即加载

如果程序加载一个持久化对象的目的是为访问他的属性,则可以采用立即加载。如果程序加载一个持久化对象的目的仅仅是为了获得他的引用,则可以采用延迟加载。

六.Hibernate在对象-关系映射问价中配置加载策略

I.类级别:

<class>元素中lazy属性的可选值为true(延迟加载)和false(立即加载);

<class>元素中的lazy属性的默认值为true

II.一对多关联级别:

<set>元素中的lazy属性的可选值为:true(延迟加载),extra(增强延迟加载)和false(立即加载);

<set>元素中的lazy属性的默认值为true

III.多对一关联级别:

<many-to-one>元素中lazy属性的可选值为:proxy(延迟加载),no-proxy(无代理延迟加载)和false(立即加载)

<many-to-one>元素中的lazy属性的默认值为proxy

在下面的一些案例中都会以员工(Emp)的部门(Dept)的例子讲解:

员工和部门是多对一的关系:

关于员工和部门的Hibernate配置就不啰嗦了:可以看我的这篇博客:http://www.cnblogs.com/heyongjun1997/p/5767187.html

I.1类级别的查询策略:

01.立即加载案例:

需求:通过Session的load()方法加载Dept对象时:

首先在Dept.hbm.xml文件中配置lazy属性为false,表示立即加载。

@Test

public void loadDept() {

// 获取Session对象

Session session = HibernateUtil.currentSession();

// 如果通过load方式加载Dept对象

Dept dept=(Dept)session.load(Dept.class, 12);

      // 关闭session

HibernateUtil.closeSession();

}

我们知道使用Load方法加载的是代理对象,只会在属性里保存一个OID,但是如果在Dept映射文件中配置了类级别的lazy为false就代表加载该对象时立即加载,也就是立即检索一道数据库,发出sql:

02.延迟加载案例:

同样是获取Dept对象,但是要把Dept.hbm.xml配置文件的类级别的lazy属性改为lazy=”true” 或者不写,应为类级别的lazy属性默认就是true,

@Test

public void loadDept() {

// 获取Session对象

Session session = HibernateUtil.currentSession();

// 如果通过load方式加载Dept对象

session.load(Dept.class, 12);

// 关闭session

HibernateUtil.closeSession();

}

此时Dept.hbm.xml配置文件的类级别的lazy属性为true,则是延迟加载,那么load方法获取的知识Dept的代理对象,所以他不会去检索数据库。

II.1一对多和多对多关联的查询策略

01.立即加载案例:

在获取部门对象的时候同时获取员工对象:

设置Dept.hbm.xml 类级别的lazy属性为false;表示立即加载:

设置<set>元素的lazy属性为false,表示在加载部门的同时立即加载员工:

   @Test

public void loadDept() {

// 获取Session对象

Session session = HibernateUtil.currentSession();

// 如果通过load方式加载Dept对象

Dept dept=(Dept)session.load(Dept.class, 12);

// 关闭session

HibernateUtil.closeSession();

}

控制台输出结果:

当你想获取一的一方(Dept)的对象同时,你也要加载多的一方(Emp)的对象,那么你要在一的一方(Emp)的<set>的节点上加上lazy="false"表示立即加载,所以在使用Load方式加载Dept对象的时候,Emp对象也会不加载出来,所以程序在运行到Dept dept=(Dept)session.load(Dept.class, 12);会发出两条sql语句:

第一条是查询部门的信息,第二条sql是根据部门编号去数据库中检索员工信息。

02.延迟加载:

如果把上面的案例<set>节点的属性lazy改为trur,或者默认不写,那么在加载Dept对象的时候,就不会再去加载Emp对象,而且只会发出一条sql,这条sql就是指检索部门的信息。

03.增强延迟加载:

当<set>元素中配置lazy的属性为extra,表明是增强延迟加载策略。

其实增强延迟加载策略与一般的延迟加载策略lazy="true"非常相似。他们主要区别在于,我们看到这个名词增强延迟加载,顾名思义就是这个策略能在进一步的帮我延迟加载这个对象,也就是代理对象的初始化时机。

演示案例:

01. 当set节点的lazy属性为true,或者不写的话(取默认值),那么执行以下语句:

@Test
    public void loadDept() {
        // 获取Session对象
        Session session = HibernateUtil.currentSession();
        // 如果通过load方式加载Dept对象
        Dept dept=(Dept)session.load(Dept.class, 12);
        //拿该部门下的员工的人数:也就是集合的大小
        dept.getEmps().size();
        // 关闭session
        HibernateUtil.closeSession();
    }

输出结果:

02. 当set节点的lazy属性为extra那么执行以下语句:

@Test
    public void loadDept() {
        // 获取Session对象
        Session session = HibernateUtil.currentSession();
        // 如果通过load方式加载Dept对象
        Dept dept=(Dept)session.load(Dept.class, 12);
        //拿该部门下的员工的人数:也就是集合的大小
        dept.getEmps().size();
        // 关闭session
        HibernateUtil.closeSession();
    }

输出结果:

III.1多对一关联的查询策略

在映射文件中,<many-to-one>元素用来设置多对一的关系,在Emp.hbm.xml文件中表明Emp类到Dept类的多对一的关联关系:

01.延迟加载

需求:获取Emp对象,但是并不去加载与之关联的Dept对象。

首先要设置<many-to-one>节点的lazy属性为proxy,表示延迟加载。

@Test
public void loadEmp() {
// 获取Session对象
Session session = HibernateUtil.currentSession();
// 如果通过load方式加载Dept对象
Emp emp=(Emp)session.get(Emp.class, 1);

//获取Dept对象,因为此时的配置文件lazy是proxy,所以是代理对象
Dept dept=emp.getDept();
// 关闭session
HibernateUtil.closeSession();
}

控制台输出结果:

结果大家可想而知:<many-to-one>节点的lazy属性为proxy,表示延迟加载。在加载Emp对象的时候会发出sql去查询数据库,但是在获取Dept对象的时候延迟加载了,所以不会发出sql。

02.无代理延迟加载:

在<many-to-one>元素中配置lazy属性为no-proxy,表示无代理延迟加载。

@Test
public void loadEmp() {
// 获取Session对象
Session session = HibernateUtil.currentSession();
// 如果通过load方式加载Dept对象
Emp emp=(Emp)session.get(Emp.class, 1);

//获取Dept对象,因为此时的配置文件lazy是proxy,所以是代理对象
Dept dept=emp.getDept();
// 关闭session
HibernateUtil.closeSession();
}

此程序在加载的Emp对象dept属性为NULL,当程序运行到第3行的时候将触发Hibernate执行查询Dept表的select语句,,从而加载Dept对象,由此可见,当lazy属性为proxy时,可以延长延迟加载Dept代理对象的时间,而lazy属性为no-proxy时,则可以避免使用由Hibernate提供的Dept代理类实例,是Hibernate对程序提供更加透明的持久化服务。

03.立即加载:

首先要设置<many-to-one>节点的lazy属性为false,表示立即加载。

@Test
public void loadEmp() {
// 获取Session对象
Session session = HibernateUtil.currentSession();
// 如果通过load方式加载Dept对象
Emp emp=(Emp)session.get(Emp.class, 1);

//获取Dept对象,因为此时的配置文件lazy是false,所以是实际对象
Dept dept=emp.getDept();
// 关闭session
HibernateUtil.closeSession();
}

控制台输出结果:

转:http://www.cnblogs.com/heyongjun1997/p/5775033.html

谢!

时间: 2024-09-30 20:08:14

Hibernate之lazy延迟加载(转)的相关文章

Hibernate中的延迟加载及fetch

Hibernate中的延迟加载 1.类级别的查询策略: lazy  :  true(默认值) false(立即加载) 2.多对一关联的查询策略: lazy:  proxy(默认值) no-proxy false 3.一对多  或多对多 lazy:true(默认值) false extra fetch:影响Hibernate针对底层SQL的生成 一.emp的映射文件的类级别设置非延迟加载 第二条会报空指针异常 二.emp的映射文件的类级别设置非延迟加载,其多对一属性也设置为非延迟加载 测试代码 三

hibernate 普通字段延迟加载无效的解决办法

关联对象的延迟加载就不说了,大家都知道. 关于普通字段的延迟加载,尤其是lob字段,若没有延迟加载,对性能影响极大.然而简单的使用 @Basic(fetch = FetchType.LAZY) 注解并没有效果.hibernate对此的解释是Lazy property loading requires buildtime bytecode instrumentation. If your persistent classes are not enhanced, Hibernate will ign

Hibernate学习之延迟加载

转自:http://www.cnblogs.com/xiaoluo501395377/p/3371776.html 在hibernate中我们知道如果要从数据库中得到一个对象,通常有两种方式,一种是通过session.get()方法,另一种就是通过session.load()方法,然后其实这两种方法在获得一个实体对象时是有区别的,在查询性能上两者是不同的. 一.load加载方式 当使用load方法来得到一个对象时,此时hibernate会使用延迟加载的机制来加载这个对象,即:当我们使用sessi

Hibernate中的延迟加载(懒加载)

什么是懒加载? Hibernate提供了一种机制,即在设置了使用懒加载的情况下,查询某一条数据时不会立即访问数据库,因此不会返回指定对象,而是返回代理对象,该代理对象并不为null,它实际上是Hibernate自动实现的指定对象所属类的子类的对象,该对象具有默认值.当要使用真正对象的属性的时候才会访问数据库,这时代理对象会自动查询数据库中对应对象的数据并返回. 这样一来降低了程序对数据库访问次数和内存使用量.下面我们通过懒加载适用情况分析它究竟是如何做到的. 懒加载的适用情况? 1.获取某一个对

(十六)Hibernate中的延迟加载

一.什么是延迟加载 为了节省Hibernate加载对象的性能节销,在Hibernate中真正需要用到这个对象时,才会发出        SQL语句来抓取这个对象.这一个过程称为延迟加载. 二.延迟加载的分类 A:实体对象的延迟加载        B:一对多|多对多的延迟加载        C:多对一|一对一的延迟加载        D:属性的延迟加载 A:实体对象的延迟加载:使用session.get()和session.load()获取对象的区别就是是否开启延迟加载. Hibernate只加载

C# lazy延迟加载

.在.NET4.0中,可以使用Lazy<T> 来实现对象的延迟初始化,从而优化系统的性能. 正如我们所知,对象的加载是需要消耗时间的,特别是对于大对象来说消耗的时间更多.lazy可以实现对象的延迟加载.延时加载,意思是对象在使用的时候创建而不是在实例化的的时候才创建. Lazy<T> 对象初始化默认是线程安全的,在多线程环境下,第一个访问 Lazy<T> 对象的 Value 属性的线程将初始化 Lazy<T> 对象,以后访问的线程都将使用第一次初始化的数据.

hibernate之4.延迟加载

延迟加载: 只有当使用以实体对象的属性(除主键属性外)时,才会发送查询语句到数据库 get不支持延迟加载 @Test public void getTest(){ Session session=null; Student student=null; try{ session=HibernateUtil.openSession(); student=(Student) session.get(Student.class, 3); System.out.println("id:"+stu

【Hibernate】Hibernate中使用延迟加载应该注意的事项

1,简介 在使用一些查询方法时,方法执行了,但是并没有立刻发送SQL语句查询数据库.而是在访问对象的getXxx方法时候才触发SQL执行加载对象数据.这种机制就称为延迟加载. 2,优点 延迟加载主要是为后续关联映射提供,避免查找无用的关联数据.可以降低数据库操作的并发率,提升内存资源使用率. 3,使用 在struts2中,session.load()和query.iterator()都使用的这种机制. 下面笔者使用session.load()方法举例: hibernate.cfg.xml 文件

hibernate的 lazy 和 fetch 一般配置

fetch 和 lazy 配置用于数据的查询 lazy 参数值常见有 false 和 true,Hibernate3 映射文件中默认lazy = true : fetch 指定了关联对象抓取的方式,参数值常见是select和join,默认是select, select方式先查询主对象,再根据关联外键,每一个对象发一个select查询,获取关联的对象,形成了n+1次查询:而join方式,是left outer join查询,主对象和关联对象用一句外键关联的sql同时查询出来,不会形成多次查询. 在