hibernate中get和load方法的区别

来源:互联网 发布:sql rowguid 编辑:程序博客网 时间:2024/05/04 16:57

getload方法的区别

getload方式是根据id取得一个记录  

1.
从返回结果上对比:
 
load
方式检索不到的话会抛出org.hibernate.ObjectNotFoundException异常
 
get
方法检索不到的话会返回
null 

2.
从检索执行机制上对比:
 
get
方法和find方法都是直接从数据库中检索
 
load方法的执行则比较复杂
 
首先查找sessionpersistent Context中是否有缓存,如果有则直接返回
 
如果没有则判断是否是lazy,如果不是直接访问数据库检索,查到记录返回,查不到抛出异常
 
如果是lazy则需要建立代理对象,对象的initialized属性为falsetarget属性为
null 
在访问获得的代理对象的属性时,检索数据库,如果找到记录则把该记录的对象复制到代理对象的target上,并将initialized=true,如果找不到就抛出异常
 

3.
根本区别说明
 
如果你使用load方法,hibernate认为该id对应的对象(数据库记录)在数据库中是一定存在的,所以它可以放心的使用,它可以放心的使用代理来延迟加载该对象。在用到对象中的其他属性数据时才查询数据库,但是万一数据库中不存在该记录,那没办法,只能抛异常。所说的load方法抛异常是指在使用该对象的数据时,数据库中不存在该数据时抛异常,而不是在创建这个对象时(注意:这就是由于延迟加载在作怪)
 

由于session中的缓存对于hibernate来说是个相当廉价的资源,所以在load时会先查一下session缓存看看该id对应的对象是否存在,不存在则创建代理。所以如果你知道该id在数据库中一定有对应记录存在就可以使用load方法来实现延迟加载。
 

对于get方法,hibernate会确认一下该id对应的数据是否存在,首先在session缓存中查找,然后在二级缓存中查找,还没有就查数据库,数据库中没有就返回null
 

对于loadget方法返回类型:虽然好多书中都这么说:“get()永远只返回实体类,但实际上这是不正确的,get方法如果在 session缓存中找到了该id对应的对象,如果刚好该对象前面是被代理过的,如被load方法使用过,或者被其他关联对象延迟加载过,那么返回的还是原先的代理对象,而不是实体类对象,如果该代理对象还没有加载实体数据(就是id以外的其他属性数据),那么它会查询二级缓存或者数据库来加载数据,但是返回的还是代理对象,只不过已经加载了实体数据。
 

get
方法首先查询session缓存,没有的话查询二级缓存,最后查询数据库;反而load方法创建时首先查询session缓存,没有就创建代理,实际使用数据时才查询二级缓存和数据库。
 

4.
简单总结
 

总之对于getload的根本区别,一句话,hibernate对于load方法认为该数据在数据库中一定存在,可以放心的使用代理来延迟加载,如果在使用过程中发现了问题,只能抛异常;而对于get方法,hibernate一定要获取到真实的数据,否则返回null

 

 

get加载对象的顺序是session cache-->second level cache-->db, 找不到返回null
load则是先返回proxy, 程序调用对象的非ID属性时,再触发session cache-->second level cache-->db的查找,找不到抛ObjectNotFoundException