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

在hibernate中我们知道如果要从数据库中得到一个对象,通常有两种方式,一种是通过session.get()方法,另一种就是通过session.load()方法,然后其实这两种方法在获得一个实体对象时是有区别的,在查询性能上两者是不同的。

一.load加载方式

当使用load方法来得到一个对象时,此时hibernate会使用延迟加载的机制来加载这个对象,即:当我们使用session.load()方法来加载一个对象时,此时并不会发出SQL语句,当前得到的这个对象其实是一个代理对象,这个代理对象只保存了实体对象的id值,只有当我们要使用这个对象,得到其它属性时,这个时候才会发出SQL语句,从数据库中去查询对象。

session = sessionfactory.openSession();

//通过load的方式加载对象时,会使用延迟加载机制,此时并不会发出sql语句,只有当我们需要使用的时候才会从数据库中去查询        
Student s = (Student)session.load(Student.class, 2);

我们看到,如果我们仅仅是通过load来加载我们的Student对象,此时控制台不会从数据库中查询出该对象,即并不会发出SQL语句。只有当我们要使用该对象时:

session = sessionfactory.openSession();
    
Student s = (Student)session.load(Student.class, 2);
System.out.println(s);

此时控制台会发出了SQL查询语句,会将该对象从数据库中查询出来。

其实第一次通过load获得的对象是一个代理对象,这个代理对象仅仅保存了id这个属性:

session = sessionfactory.openSession();
            
//通过load的方式加载对象时,会使用延迟加载机制,此时得到的对象其实是一个代理对象,该代理对象里面仅仅只有id这个属性
//而且即使数据库中没有Id=10的字段,也不会报错
Student s = (Student)session.load(Student.class, 10);
System.out.println(s.getId());

如果我们只打印出这个对象的id值时,此时控制台会打印出该id值,但是同样不会发出SQL语句去从数据库中去查询,即使数据库中没有Id=10的字段,也不会报错。这就印证了我们的这个Student对象仅仅是一个保存了id的代理对象,但如果我需要打印出Student对象的其他属性值时,这个时候会不会发出SQL语句呢?答案是肯定会发出的。


那第二次执行load是怎么样的呢?

session = sessionfactory.openSession();
            
Student s1 = (Student)session.load(Student.class, 2);
System.out.println(s1.getAge());
Student s2 = (Student)session.load(Student.class, 2);
System.out.println(s1==s2);//输出true

下图显示了第一次执行Session的load()方法的过程。load方法先到Session缓存中查找OID为2的Student对象,由于Session缓存还不存在,于是load()创建一个OID为2的对象并返回,只有当要用到这个对象的时候,才到数据库中去查找,并将其保存在Session缓存中。

下图显示了第二次执行Session的load()方法的过程。load()方法先到Session缓存中查找OID为2的Student对象,因为已经存在这样的Student对象了,就直接返回该Student对象。因此在上面的程序代码中,变量s1和变量s2引用的是同一个Student对象。

二.get加载方式

相对于load的延迟加载方式,get就直接的多,当我们使用session.get()方法来得到一个对象时,不管我们使不使用这个对象,此时都会发出SQL语句去从数据库中查询出来。

session = sessionfactory.openSession();
            
Student s1 = (Student)session.get(Student.class, 2);
Student s2 = (Student)session.get(Student.class, 2);

System.out.println(s1==s2);

下图显示了第一次执行Session的get()方法的过程。get方法先到Session缓存中查找OID为2的Student对象,由于Session缓存还不存在,于是到数据库中去查找,并将其保存在Session缓存中。

下图显示了第二次执行Session的get()方法的过程。get()方法第二次执行的过程与load()方法一致,先到Session缓存中查找OID为2的Student对象,因为已经存在这样的Student对象了,就直接返回该Student对象。因此在上面的程序代码中,变量s1和变量s2引用的也是同一个Student对象。

三.使用get和load时的要注意的地方

1、load()方法的这种加载方式称为懒加载,我们可以通过修改配置文件取消懒加载

<class  name="Student" lazy="false" table="student">

2、如果确定数据库中有这个对象就用load(),不确定就用get()(这样效率高)

3、如果使用get方式来加载对象,当我们试图得到一个id不存在的对象时,此时会报NullPointException的异常。

session = sessionfactory.openSession();
            
Student s = (Student)session.get(Student.class, 10);        
System.out.println(s.getName());    

这是因为通过get方式我们会去数据库中查询出该对象,但是这个id值不存在,所以此时user对象是null,所以就会报NullPointException的异常了。

4、如果使用load方式来加载对象,当我们试图得到一个id不存在的对象时,此时会报ObjectNotFoundException异常。

session = sessionfactory.openSession();
            
Student s = (Student)session.load(Student.class, 10);        
System.out.println(s.getName());

这是因为使用load时,此时的Student对象是一个代理对象,仅仅保存了当前的这个id值,当我们试图得到该对象的name属性时,这个属性其实是不存在的,所以就会报出ObjectNotFoundException这个异常了。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值