hibernate延迟加载
时间:2010-07-18 来源:foohsinglong
在hibernate应用中,有时候只访问指定的持久化对象,在加载该对象时没有必要加载其关联的对象,以免浪费内存空间,但若想在Session关闭后正常访问该对象,必须在加载该对象时,一起加载该对象的关联对象,为了满足检索持久化对象的不同情况的需求,开发灵活高效的hibernate应用,hibernate提供的三种检索策略:立即检索、延迟检索(加载)和迫切连接检索。
在hibernate 3中为了提高效率、节省内存,默认的加载模式是延迟加载,也就是说没有在ORM配置文件中显示指定lazy参数的参数的话,laze的默认值是true,hibernate将默认为延迟加载,例如:有2个持久化对象:TTask和Tusers
其中TUsers是TTask的一个多对一的关联对象,TTask.hbm.xml中配置内容为:
测试程序段:
……
这是因为在运行到(1)出的时候没有查询TTask的关联对象TUser,之后Session关闭, 运行到(2)处时,由于session关闭,所以就抛出了异常。
延迟加载作为hibernate默认检索策略,所谓延迟加载就是在等到访问的时候才装载和初始化指定的对象,如果在Session实例关闭之前没有初始化延迟检索出来的对象session实例关闭后再访问时,就会抛出异常,因此在访问指定的对象前应确保该对象被成功实例化。
延迟加载主要可以用在:
Hibernate3.x,lazy默认是打开的
Hibernate Lazy策略使用:
<class>标签,可以取值true/false
<property>标签,可以取值true/false
<set><list>标签,可以取值true/false/extra
<one-to-one> <many-to-one>单端关联,可以取值:false/proxy/noproxy
load支持lazy,get不支持,无论持久化类设置与否,都会立即检索。
(1)class标签
class上的Lazy,只对普通属性起作用,对集合和<one-to-one> <many-to-one>不起作用
即:如果一个类里有Set集合成员变量和需要用<one-to-one> <many-to-one>来映射的成员变量,即使在class里设置lazy为false,他们在使用之前也不会加载。需要单独设置lazy,也就是lazy的另外两个用法。
如果在使用load加载了某个对象,但是没有使用,即没有发出SQL语句,就把session给关了,此时,如果再使用这个对象就会抛出LazyInitalizationException。因为对象并没有加载上来,因为使用了lazy策略,只是加载了类的一个代理子类。
(2)集合
默认为true情况下,只要使用到了Set对象,就会把整个set全部查询出来。
false情况下,不使用Lazy,查询Lazy所属的对象时,set就会被查询上来。需要类增强工具。
extra情况下,比较智能,根据查询的内容,生成不同的SQL语句。效率会高一些。
(3)单端关联
false:不使用Lazy。
noproxy:需要类增强工具
(4)property
true:较少使用,如果某个字段是内容量比较大,可以设置为true,在真正使用的时候再加载。可以提高效率。
在hibernate 3中为了提高效率、节省内存,默认的加载模式是延迟加载,也就是说没有在ORM配置文件中显示指定lazy参数的参数的话,laze的默认值是true,hibernate将默认为延迟加载,例如:有2个持久化对象:TTask和Tusers
其中TUsers是TTask的一个多对一的关联对象,TTask.hbm.xml中配置内容为:
<many-to-one name="TUsersByUserId" class="DAO.TUsers" fetch="select"> <column name="user_id" length="20" /> </many-to-one>
测试程序段:
public void test(){ Session session=HibernateSessionFactory.getSession(); Query q=session.createQuery("from TTask"); List list=q.list(); (1) HibernateSessionFactory.closeSession(); for(int i=0;i<list.size();i++){ TTask t=(TTask)list.get(i); System.out.println(t.getTUsersByCarryId().getUsername());(2) } }会造成如下异常:
Exception in thread "main" org.hibernate.LazyInitializationException: could not initialize proxy - no Session
……
这是因为在运行到(1)出的时候没有查询TTask的关联对象TUser,之后Session关闭, 运行到(2)处时,由于session关闭,所以就抛出了异常。
延迟加载作为hibernate默认检索策略,所谓延迟加载就是在等到访问的时候才装载和初始化指定的对象,如果在Session实例关闭之前没有初始化延迟检索出来的对象session实例关闭后再访问时,就会抛出异常,因此在访问指定的对象前应确保该对象被成功实例化。
延迟加载主要可以用在:
Hibernate3.x,lazy默认是打开的
Hibernate Lazy策略使用:
<class>标签,可以取值true/false
<property>标签,可以取值true/false
<set><list>标签,可以取值true/false/extra
<one-to-one> <many-to-one>单端关联,可以取值:false/proxy/noproxy
load支持lazy,get不支持,无论持久化类设置与否,都会立即检索。
(1)class标签
<class name="className" lazy="true/alse">默认为true
class上的Lazy,只对普通属性起作用,对集合和<one-to-one> <many-to-one>不起作用
即:如果一个类里有Set集合成员变量和需要用<one-to-one> <many-to-one>来映射的成员变量,即使在class里设置lazy为false,他们在使用之前也不会加载。需要单独设置lazy,也就是lazy的另外两个用法。
如果在使用load加载了某个对象,但是没有使用,即没有发出SQL语句,就把session给关了,此时,如果再使用这个对象就会抛出LazyInitalizationException。因为对象并没有加载上来,因为使用了lazy策略,只是加载了类的一个代理子类。
(2)集合
<set name="name" lazy="true/false/extra" >默认为true
默认为true情况下,只要使用到了Set对象,就会把整个set全部查询出来。
false情况下,不使用Lazy,查询Lazy所属的对象时,set就会被查询上来。需要类增强工具。
extra情况下,比较智能,根据查询的内容,生成不同的SQL语句。效率会高一些。
(3)单端关联
<many-to-one name="name" lazy="false/proxy/noproxy">默认为proxy
false:不使用Lazy。
noproxy:需要类增强工具
(4)property
<property name="name" lazy="true/false"/>默认false
true:较少使用,如果某个字段是内容量比较大,可以设置为true,在真正使用的时候再加载。可以提高效率。
相关阅读 更多 +