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

Stella981
• 阅读 393

Hibernate中根据Id单条查询获取对象的方式有两种,分别是get()和load(),来看一下这两种方式的区别。

1. get()

使用get()来根据ID进行单条查询:

User user=session.get(User.class, "1");

当get()方法被调用的时候就会立即发出SQL语句:

Hibernate:
    select
        user0_.ID as ID1_1_0_,
        user0_.CREATETIME as CREATETI2_1_0_,
        user0_.UPDATETIME as UPDATETI3_1_0_,
        user0_.USERNAME as USERNAME4_1_0_,
        user0_.PASSWD as PASSWD5_1_0_
    from
        USER user0_
    where
        user0_.ID=?

并且返回的对象也是实际的对象:

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

使用get()和普通的单条查询并没有多大的区别。

2. load()

当使用load()进行查询的时候情况就变得很不一样了:

User user=session.load(User.class, "1");

当调用load()方法的时候会返回一个目标对象的代理对象,在这个代理对象中只存储了目标对象的ID值,只有当调用除ID值以外的属性值的时候才会发出SQL查询的。

返回值:

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

在handler中有一个属性叫做target,保存着被代理的对象:

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

现在这个位置还是空的呢。

当我们尝试下面的代码时:

User user=session.load(User.class, "1");
System.out.println(user.getId());

因为我们只访问了ID属性,这个在代理对象中是已经存在的了,所以并不需要再去数据库中查询,因此并不会发出SQL查询语句。

当使用到除ID以外的属性的时候,会发出SQL查询语句,比如尝试执行下面的代码:

User user=session.load(User.class, "1");
System.out.println(user.getUsername());

会发现控制台打印了SQL查询语句:

Hibernate:
    select
        user0_.ID as ID1_1_0_,
        user0_.CREATETIME as CREATETI2_1_0_,
        user0_.UPDATETIME as UPDATETI3_1_0_,
        user0_.USERNAME as USERNAME4_1_0_,
        user0_.PASSWD as PASSWD5_1_0_
    from
        USER user0_
    where
        user0_.ID=?

这个时候再看代理对象的话,会发现target已经被填充上了:

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

3. Exception

下面的代码会报一个空指针异常:

User user=session.get(User.class, "foobar");
System.out.println(user.getUsername());

上面的这段代码抛出了空指针异常 NPE:

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

这个是很容易理解的,因为没有查询到的就空指针了嘛。

而下面的这段代码则会报一个ObjectNotFoundException异常:

User user=session.load(User.class, "foobar");
System.out.println(user.getUsername());

抛出了ObjectNotFoundException异常:

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

这个就有点奇怪了,这个是因为我们使用load()的时候返回的是一个代理对象,因为这个时候还没有进行查询,所以我们并没有办法确定要查询的对象到底存在不存在,所以使用load()查询的返回值是永远不会为空的,但是呢,当我们试图访问被代理的真实对象的时候,因为这个对象并不存在,所以就抛出了一个ObjectNotFoundException。

还有一种说法是get()是用于不确定对象是否真的存在的情况,所以在查询出来后可以先进行一个空指针判断,而load()方法用于对象一定存在的情况下,不然等会儿使用的时候就可能会抛出ObjectNotFoundException了。

再来看下面这段代码:

@Test
public void test_001(){
     
    SessionFactory sessionFactory=new Configuration().configure().buildSessionFactory();
    Session session=sessionFactory.openSession();
     
    User user=null;
     
    try {
        session.beginTransaction();
         
        user=session.load(User.class, "foobar");
         
        session.getTransaction().commit();
    } catch (Exception e) {
        session.getTransaction().rollback();
        e.printStackTrace();
    }finally{
        try {
            session.close();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
     
    System.out.println(user.getUsername());
     
}

抛出了一个LazyInitializationException:

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

这个是因为我们使用load()查询出来的对象只有在调用完非ID属性的时候才会去查询数据填充进来,但是查询数据的时候是需要依赖产生这个代理对象的那个Session去查询的,当我们将Session关闭后,再试图去访问非ID属性,它正打算拿着自己依赖的Session去数据库查询,一看Session竟然被关闭了,得,干脆抛出一个LazyInitializationException好了。

解决办法就是在查询的时候思考一下,这个对象时候需要在Session关闭之后还能使用得到呢?如果是的话,那么我们使用get()来查询,或者手动调用空访问一个非ID属性让它把数据回填上先。

4. 缓存

get()和load()都会使用缓存,都是首先从一级缓存Session中查找,当找不到的时候再去二级缓存中查找,当查询不到的时候get()返回的是null,而load()则返回代理对象。

来看下面这段代码:

User user1=session.get(User.class, "1");
User user2=session.load(User.class, "1");

来看一下这两条执行过后这两个变量的值:

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

会发现两个都是真实对象,连load()返回的也是真实对象,并且它们引用的还是同一块对象。

这个是因为get()查询出来ID为1的对象后会将其放入到缓存中,而load()再去查询的时候它会先去缓存中查找,如果缓存中没有的话才会返回代理对象,但是当缓存中已经存在的话就直接将真实对象返回来了。

5. 对比总结

返回值:

get()返回的是查询出来的实体对象,而load()查询出来的是一个目标实体的代理对象。

查询时机:

get()在调用的时候就立即发出SQL语句查询,而load()在访问非ID属性的时候才会发出查询语句并且将被代理对象target填充上,但是如果这个动作发生在Session被关闭后的话就会抛出LazyInitializationException。

查询结果为空时:

get()抛出NullPointerException

load()抛出ObjectNotFoundException

文章转载自:https://www.cnblogs.com/cc11001100/p/6883790.html

点赞
收藏
评论区
推荐文章
blmius blmius
1年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Jacquelyn38 Jacquelyn38
2年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
helloworld_34035044 helloworld_34035044
11个月前
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
Stella981 Stella981
1年前
JS 对象数组Array 根据对象object key的值排序sort,很风骚哦
有个js对象数组varary\{id:1,name:"b"},{id:2,name:"b"}\需求是根据name或者id的值来排序,这里有个风骚的函数函数定义:function keysrt(key,desc) {  return function(a,b){    return desc ? ~~(ak
Stella981 Stella981
1年前
GitHub Actions
使用GitHubActions发布版本时,获取触发的tag作为发布的版本号.方式一通过step获取tag,在需要使用的地方使用steps.get_version.outputs.VERSION,其中get_version是step的id.name:Release
Stella981 Stella981
1年前
From表单提交的几种方式
<body<formaction"https://my.oschina.net/u/3285916"method"get"name"formOne"id"formId"name:<inputtype"text"name"name"pwd:<inputtyp
Stella981 Stella981
1年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable
Wesley13 Wesley13
1年前
Oracle一张表中实现对一个字段不同值和总值的统计(多个count)
需求:统计WAIT\_ORDER表中的工单总数、未处理工单总数、已完成工单总数、未完成工单总数。表结构:为了举例子方便,WAIT\_ORDER表只有两个字段,分别是ID、STATUS,其中STATUS为工单的状态。1表示未处理,2表示已完成,3表示未完成总数。 SQL:  1.SELECT   2
Wesley13 Wesley13
1年前
Hibernate中get和load方法的区别以及close()、clear()、evict()
下边详细说一下get和load的不同,因为有些时候为了对比也会把find加进来。1.从返回结果上对比:load方式检索不到的话会抛出org.hibernate.ObjectNotFoundException异常get方法检索不到的话会返回null2.从检索执行机制上对比:get方法和find方法都是直接从数据库中检索而load
Wesley13 Wesley13
1年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_