最近在学习SHH框架中的hibernate,对Session的get和load方法,有点混不清楚,不知道区别在哪,或者对它们的区别感触不深。所以百度了一下,结果问题来了。百度的结果和实际测试的结果出入很大。主要是对get方法的说法跟实际运行的结果不一致。

先说一下观点吧:

  • get不支持lazy,load支持lazy;
  • 数据不存在时,get返回null,load则抛出ObjectNotFoundException异常。
  • load方法可以返回实体的代理类实例,而get方法直接读取数据库,所以直接返回实体类(get的这个说法是错误的)
 
      对于第一条,相信大家都没有太多的疑问。我这里给个例子稍作解释:lazy意味着用的时候才去执行sql语句。
  1. User user = (User)session.load(User.class,"4028981b41174a690141174a6c6d0003");

这句代码不会去执行数据库查询,只有用到user时才会去执行数据库查询。所以不会立即生成sql语句。

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

而上面这句代码则会立即去执行数据库查询(如果缓存中没有实例)。

而后面的问题要想说明白,首先得了解一个问题——Session加载实体对象的过程:

首先,Hibernate中维持了两级缓存。第一级缓存由Session实例维护,它是属于事务范围的缓存。其中保持了Session当前所有关联实体的数据,也称为内部缓存。而第二级缓存则存在于SessionFactory层次,它是属于进程范围或群集范围的缓存,由当前所有由本SessionFactory构造的Session实例共享。

出于性能考虑,避免无谓的数据库访问,Session在调用数据库查询功能之前,会先在缓存中进行查询。首先在第一级缓存(内部缓存)中,通过实体类型和id进行查找,如果第一级缓存查找命中,且数据状态合法,则直接返回。然后,Session会在当前“NonExists”记录中进行查找,如果“NonExists”记录中存在同样的查询条件,则返回null。 “NonExists” 记录了当前Session实例在之前所有查询操作中,未能查询到有效数据的查询条件(相当于一个查询黑名单列表)。如此一来,如果Session中一个无效的查询条件重复出现,即可迅速作出判断,从而获得最佳的性能表现。

 
      对于load方法而言,首先查找内部缓存,如果命中,则返回实例,如果内部缓存中未发现有效数据,则查询第二级缓存,如果第二级缓存命中,则返回。若二级缓存中依旧未发现有效数据,则发起数据库查询操作(Select SQL)。如果查询到,则返回实体类的代理对象,若经过查询未发现对应记录,则将此次查询的信息在“NonExists” 中加以记录,并抛出ObjectNotFoundException异常。
 
      对于get方法而言,许多书上、网络博客里都说错了。get方法同样是先查找内部缓存,如果命中,则返回,否则发起数据库查询操作,如果查询到,则返回实体类的对象,若经过查询未发现对应记录,则将此次查询的信息在“NonExists” 中加以记录,并返回null。所以网络上说的“当他人修改了数据后,用load可能读取不到最新的数据,而get肯定可以读取到最新修改的数据”的说法也是不成立的。
 
      这也就意味着,get方法获取到的并不一定是实体类对象,load方法也不一定是返回实体代理类对象。
 
      以上的观点都是我通过测试得出来的,有代码有图有真相呀:
  1. package com.bjpowernode.hibernate;
  2. import java.util.Date;
  3. import junit.framework.TestCase;
  4. import org.hibernate.ObjectNotFoundException;
  5. import org.hibernate.Session;
  6. import org.hibernate.Transaction;
  7. /**
  8. * Session测试类
  9. *
  10. * @author Longxuan
  11. *
  12. */
  13. public class SessionTest extends TestCase {
  14. public void testEquals() {
  15. Session session = null;
  16. try {
  17. //获取Session
  18. session = HibernateUtils.getSession();
  19. // 开启事务
  20. session.beginTransaction();
  21. System.out.println("\n\n\n\n");
  22. try {
  23. // 验证查不到数据时,get返回null,load抛ObjectNotFoundException异常
  24. System.out.println(session.get(User.class, "123"));
  25. System.out.println(session.load(User.class, "123"));
  26. } catch (ObjectNotFoundException e) {
  27. System.out.println("load方法抛出ObjectNotFoundException异常");
  28. }
  29. System.out.println("\n\n");
  30. // 验证load返回实体类对象,而非代理对象
  31. {
  32. User user1 = (User) session.get(User.class,"4028981b41174a690141174a6c6d0003");
  33. User user2 = (User) session.load(User.class,"4028981b41174a690141174a6c6d0003");
  34. System.out.println("user1:" + user1.getClass().getSimpleName());
  35. System.out.println("user2:" + user2.getClass().getSimpleName());
  36. System.out.println("user1与 user2是否为同一对象:" + user1.equals(user2));
  37. }
  38. System.out.println("\n\n");
  39. session.clear();//清除Session
  40. // 验证get也可以返回代理类对象,而并不一定返回实体类对象
  41. // 同时验证了get方法先查找缓存(如果没有输出sql语句,则说明get查找了缓存)
  42. {
  43. User user3 = (User) session.load(User.class,"4028981b41174a690141174a6c6d0003");
  44. User user4 = (User) session.get(User.class,"4028981b41174a690141174a6c6d0003");
  45. System.out.println("user3:" + user3.getClass().getSimpleName());
  46. System.out.println("user4:" + user4.getClass().getSimpleName());
  47. System.out.println("user3与 user4是否为同一对象:" + user3.equals(user4));
  48. }
  49. session.getTransaction().commit();
  50. } catch (Exception e) {
  51. e.printStackTrace();
  52. session.getTransaction().rollback();
  53. } finally {
  54. HibernateUtils.closeSession(session);
  55. }
  56. }
  57. }

运行结果图:

 

      还有一个有趣的现象:
  1. User user5 = (User)session.load(User.class, "123");
  2. System.out.println(user5.getId());

运行结果直接输出 123

      从结果中也可以看出,前2句代码,不会去执行数据库操作。因为load后会在hibernate的一级缓存里存放一个map对象,该map的key就是Id的值,但是当你getId()时,它会去一级缓存里拿map的key值,正好找到了,所以不会再去执行数据库查询。也不会报任何错。就有了以上的结果。

Hibernate中Session之get和load方法的真正区别的更多相关文章

  1. Hibernate中Session.get()方法和load()方法的详细比较

    一.get方法和load方法的简易理解  (1)get()方法直接返回实体类,如果查不到数据则返回null.load()会返回一个实体代理对象(当前这个对象可以自动转化为实体对象),但当代理对象被调用 ...

  2. hibernate中session的get和load方法的区别和联系:

    1. get:及时加载,调用到get方法时立即向数据库查询(在没有session缓存的请况). 2. load:默认使用懒加载,当用到数据的时候才向数据库查询(在没有session缓存的请况). 3. ...

  3. Hibernate中Session的get和load

    hibernate中Session接口提供的get()和load()方法都是用来获取一个实体对象,在使用方式和查询性能上有一些区别.测试版本:hibernate 4.2.0. get Session接 ...

  4. [转]Hibernate中Session的get和load

    hibernate中Session接口提供的get()和load()方法都是用来获取一个实体对象,在使用方式和查询性能上有一些区别.测试版本:hibernate 4.2.0. get Session接 ...

  5. Hibernate的Session的get()和load()方法区别

    hibernate中Session接口提供的get()和load()方法都是用来获取一个实体对象,在使用方式和查询性能上有一些区别. get Session接口提供了4个重载的get方法,分别通过“持 ...

  6. Hibernate中Session.get()/load()之区别

    原文链接http://sunxin1001.iteye.com/blog/292090 Session.load/get方法均可以根据指定的实体类和id从数据库读取记录,并返回与之对应的实体对象.其区 ...

  7. hibernate中session

    hibernate中的session是一级缓存,可以理解为进程级的缓存.在进程运行期间一直存在. session可以理解为一个可以操作数据库的对象 具体如何操作数据库? session中有方法, 如果 ...

  8. hibernate中session的获取使用以及其他注意事项

    hibernate中session的获取使用以及其他注意事项 前言:工作时,在同时使用Hibernate的getSession().getHibernateTemplate()获取Session后进行 ...

  9. Hibernate中Session与本地线程绑定

    ------------------siwuxie095 Hibernate 中 Session 与本地线程绑定 1.Session 类似于 JDBC 的连接 Connection 2.Session ...

随机推荐

  1. 《深入实践Spring Boot》阅读笔记之一:基础应用开发

    上上篇「1718总结与计划」中提到,18年要对部分项目拆分,进行服务化,并对代码进行重构.公司技术委员会也推荐使用spring boot,之前在各个技术网站中也了解过,它可以大大简化spring配置和 ...

  2. Apollo单向SSL认证(1)

    参考链接:https://www.cnblogs.com/benwu/articles/4891758.html keytool -genkey -alias mybroker -keyalg RSA ...

  3. 【52ABP实战教程】00-- ASP.NET CORE系列介绍

    为什么是.net core? 记得在半年前.NET CORE刚刚出了1.0,当时有朋友推荐我使用的时候,个人觉得还不成熟. 现在.NET Core已经到了2.0,.NET Standard 2.0 添 ...

  4. jprofiler配置

    cataline.sh JAVA_OPTS="$JAVA_OPTS -agentlib:jprofilerti=port=8849"JAVA_OPTS="$JAVA_OP ...

  5. MVC、MVP以及MVVM分析

    网上现在MVC.MVP以及MVVM的讲解一搜一箩筐,根据了网上大多数的文章,根据我的思考习惯进行了总结. MVC介绍及分析: 各层的职责如下所示: Models: 数据层,负责数据的处理和获取的数据接 ...

  6. 关于block的循环引用的问题

    在block的循环引用的问题上我们都知道如果在block内部修改外部的变量的时候,要加__block以防止循环引用的问题,但是如果block是当前对象的一个属性的时候,要修改当前对象的一个属性的时候就 ...

  7. Android:ImageView控件显示图片

    1)android显示图片可以使用imageView来呈现,而且也可以通过ImageButton来实现给button添加图片. 2)在创建一个ImageView后,显示图片绑定元素是:android: ...

  8. Checkbutton

    #tkinter之Checkbutton篇 #Checkbutton又称为多选按钮,可以表示两种状态,On和Off,可以设置回调函数,每当点击此按钮时回调函数被调用. 1.一个简单的Checkbutt ...

  9. 使用生成器把Kafka写入速度提高1000倍

    title: 使用生成器把Kafka写入速度提高1000倍 toc: true comment: true date: 2018-04-13 21:35:09 tags: ['Python', '经验 ...

  10. reportng优化

    本来呢,我是看到报告中没有输出@Test的description 的属性,想优化一下,没想到在找reportng的源码的时候,发现一个大神也优化了reportNG,他优化了下面几个内容: 1).测试结 ...