Hibernate之延迟加载(懒加载)
来源:互联网 发布:钱箱控制软件 编辑:程序博客网 时间:2024/05/21 15:04
定义:
延迟加载机制是为了避免一些无谓的性能开销而提出来的,所谓延迟加载就是当在真正需要数据的时候,才真正执行数据加载操作。在Hibernate中提供了对实体对象的延迟加载以及对集合的延迟加载,另外在Hibernate3中还提供了对属性的延迟加载。
A、实体对象的延迟加载:
如果想对实体对象使用延迟加载,必须要在实体的映射配置文件中进行相应的配置,如下所示:
<hibernate-mapping> <class name=”com.neusoft.entity.User” table=”user” lazy=”true”> …… </class></hibernate-mapping>
通过将class的lazy属性设置为true,来开启实体的延迟加载特性。如果我们运行下面的代码:
User user=(User)session.load(User.class,”1”);(1)
System.out.println(user.getName());(2)
当运行到(1)处时,Hibernate并没有发起对数据的查询。记得前面我曾讲过session.load()方法,会返回实体对象的代理类对象,这里所返回的对象类型就是User对象的代理类对象。在Hibernate中通过使用CGLIB,来实现动态构造一个目标对象的代理类对象,并且在代理类对象中包含目标对象的所有属性和方法,而且所有属性均被赋值为null。通过调试器显示的内存快照,我们可以看出此时真正的User对象,是包含在代理对象的CGLIB$CALBACK_0.target属性中。
当代码运行到(2)处时,此时调用user.getName()方法,这时通过CGLIB赋予的回调机制,当调用该方法时,Hibernate会首先检查CGLIB$CALBACK_0.target
属性是否为null,如果不为空,则调用目标对象的getName方法,如果为空,则会发起数据库查询,生成类似这样的SQL语句:select * from user where id=’1’;来查询数据,并构造目标对象,并且将它赋值到CGLIB$CALBACK_0.target
属性中。
这样,通过一个中间代理对象,Hibernate实现了实体的延迟加载,只有当用户真正发起获得实体对象属性的动作时,才真正会发起数据库查询操作。所以实体的延迟加载是用通过中间代理类完成的,所以只有session.load()方法才会利用实体延迟加载,因为只有session.load()方法才会返回实体类的代理类对象
集合类型的延迟加载
<hibernate-mapping> <class name=”com.neusoft.entity.User” table=”user”>…..<set name=”addresses” table=”address” lazy=”true” inverse=”true”><key column=”user_id”/> <one-to-many class=”com.neusoft.entity.Arrderss”/></set> </class> </hibernate-mapping>
通过将元素的lazy属性设置为true来开启集合类型的延迟加载特性。我们看下面的代码:
User user=(User)session.load(User.class,”1”);Collection addset=user.getAddresses(); (1)Iterator it=addset.iterator(); (2)while(it.hasNext()){ Address address=(Address)it.next(); System.out.println(address.getAddress());}
当程序执行到(1)处时,这时并不会发起对关联数据的查询来加载关联数据,只有运行到(2)处时,真正的数据读取操作才会开始,这时Hibernate会根据缓存中符合条件的数据索引,来查找符合条件的实体对象。
属性延迟加载:
在Hibernate3中,引入了一种新的特性——属性的延迟加载,这个机制又为获取高性能查询提供了有力的工具。在前面我们讲大数据对象读取时,在User对象中有一个resume字段,该字段是一个java.sql.Clob类型,包含了用户的简历信息,当我们加载该对象时,我们不得不每一次都要加载这个字段,而不论我们是否真的需要它,而且这种大数据对象的读取本身会带来很大的性能开销。在Hibernate2中,我们只有通过我们前面讲过的面性能的粒度细分,来分解User类,来解决这个问题(请参照那一节的论述),但是在Hibernate3中,我们可以通过属性延迟加载机制,来使我们获得只有当我们真正需要操作这个字段时,才去读取这个字段数据的能力,为此我们必须如下配置我们的实体类:
<hibernate-mapping> <class name=”com.neusoft.entity.User” table=”user”> <property name=”resume” type=”java.sql.Clob” column=”resume” lazy=”true”/> </class></hibernate-mapping>
通过对元素的lazy属性设置true来开启属性的延迟加载,在Hibernate3中为了实现属性的延迟加载,使用了类增强器来对实体类的Class文件进行强化处理,通过增强器的增强,将CGLIB的回调机制逻辑,加入实体类,这里我们可以看出属性的延迟加载,还是通过CGLIB来实现的。CGLIB是Apache的一个开源工程,这个类库可以操纵java类的字节码,根据字节码来动态构造符合要求的类对象。根据上面的配置我们运行下面的代码:
String sql=”from User user where user.name=’zx’ ”;Query query=session.createQuery(sql); (1)List list=query.list();for(int i=0;i<list.size();i++){ User user=(User)list.get(i); System.out.println(user.getName()); System.out.println(user.getResume()); (2)}
当执行到(1)处时,会生成类似如下的SQL语句:
Select id,age,name from user where name=’zx’;
这时Hibernate会检索User实体中所有非延迟加载属性对应的字段数据,当执行到(2)处时,会生成类似如下的SQL语句:
Select resume from user where id=’1’;
这时会发起对resume字段数据真正的读取操作。
参考博客:
http://blog.csdn.net/lhx222/article/details/1680610
http://blog.sina.com.cn/s/blog_67fdef9001010ktt.html
http://www.cnblogs.com/lanzhi/p/6469527.html(重点)
- Hibernate之延迟加载(懒加载)
- Hibernate延迟加载之属性延迟加载
- hibernate延迟加载(懒加载)详解
- hibernate延迟加载(懒加载)详解
- hibernate延迟加载(懒加载)详解
- hibernate延迟加载(懒加载)详解
- hibernate延迟加载(懒加载)详解
- hibernate延迟加载(懒加载)详解
- hibernate延迟加载(懒加载)详解
- Hibernate(一) 之 延迟加载
- Hibernate学习之抓取策略与懒(延迟)加载
- hibernate之4.延迟加载
- Hibernate之lazy延迟加载
- Hibernate 延迟加载(懒加载)与抓取策略
- Hibernate性能优化 --- 延迟加载(懒加载)
- Hibernate性能优化 --- 延迟加载(懒加载)
- Hibernate懒加载/延迟加载机制总结
- Hibernate中的延迟加载(懒加载)
- js中函数的三种定义方式、函数声明、函数同名重复、函数删除、
- ip地址正则表达式(两种方式)
- test
- 计算机组成原理与体系结构---数据的表示
- Java实现二叉树的前序、中序、后序、层序遍历(非递归方法)
- Hibernate之延迟加载(懒加载)
- 进程同步和线程同步
- python-输入、输出、类型转换、if、运算符、流程控制
- 深度学习----初窥门径笔记
- SpringCloud(第 004 篇)用户服务类(添加服务注册,将用户微服务注册到 EurekaServer 中)
- sql 50题
- html help workshop v4.73 中文版 附使用教程
- 17西安网络赛 规律题 Maximum Flow
- yii2 restful同一个控制器设置允许认证不允许认证方法,跨域设置