Hibernate 中 load() 和 get() 的区别

时间:2024-01-01 18:39:09

get 和 load 方式都是是根据 id 取得一个记录。
下边详细说一下 get 和 load 的不同,因为有些时候为了对比也会把 find 加进来。

1.从返回结果上对比:

load 方式检索不到的话会抛出 org.hibernate.ObjectNotFoundException 异常
get 方法检索不到的话会返回 null

2.从检索执行机制上对比:

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

3.根本区别说明

■ 如果你使用 load 方法,hibernate 认为该 id 对应的对象(数据库记录)在数据库中是一定存在的,所以它可以放心的使用,它可以放心的使用代理来延迟加载该对象。在用到对象中的其他属性数据时才查询数据库,但是万一数据库中不存在该记录,那没办法,只能抛异常。所说的 load 方法抛异常是指在使用该对象的数据时,数据库中不存在该数据时抛异常,而不是在创建这个对象时(注意:这就是由于“延迟加载”在作怪)。由于 session 中的缓存对于 hibernate 来说是个相当廉价的资源,所以在 load 时会先查一下 session 缓存看看该 id 对应的对象是否存在,不存在则创建代理。所以如果你知道该 id 在数据库中一定有对应记录存在就可以使用 load 方法来实现延迟加载。

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

对于load和get方法返回类型:

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

4.简单总结

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

【转自load和get的区别