1. 对于Hibernate get方法,Hibernate会确认一下该id对应的数据是否存在,首先在session缓存中查找,而后在二级缓存中查找,尚未就查询数据库,数据库中没有就返回null。这个相对比较简单,也没有太大的争议。主要要说明的一点就是在这个版本中get方法也会查找二级缓存!java
2. Hibernate load方法加载实体对象的时候,根据映射文件上类级别的lazy属性的配置(默认为true),分状况讨论:sql
(1)若为true,则首先在Session缓存中查找,看看该id对应的对象是否存在,不存在则使用延迟加载,返回实体的代理类对象(该代理类为实体类的子类,由CGLIB动态生成)。等到具体使用该对象(除获取OID之外)的时候,再查询二级缓存和数据库,若仍没发现符合条件的记录,则会抛出一个ObjectNotFoundException。数据库
(2)若为false,就跟Hibernate get方法查找顺序同样,只是最终若没发现符合条件的记录,则会抛出一个ObjectNotFoundException。缓存
事例解析:session
一.load加载方式ide
当使用load方法来获得一个对象时,此时hibernate会使用延迟加载的机制来加载这个对象,即:当咱们使用session.load()方法来加载一个对象时,此时并不会发出sql语句,当前获得的这个对象实际上是一个代理对象,这个代理对象只保存了实体对象的id值,只有当咱们要使用这个对象,获得其它属性时,这个时候才会发出sql语句,从数据库中去查询咱们的对象。性能
session = HibernateUtil.openSession(); /* * 经过load的方式加载对象时,会使用延迟加载机制,此时并不会发出sql语句,只有当咱们须要使用的时候才会从数据库中去查询 */ User user = (User)session.load(User.class, 2);
咱们看到,若是咱们仅仅是经过load来加载咱们的User对象,此时从控制台咱们会发现并不会从数据库中查询出该对象,即并不会发出sql语句,但若是咱们要使用该对象时:spa
session = HibernateUtil.openSession(); User user = (User)session.load(User.class, 2); System.out.println(user);
此时咱们看到控制台会发出了sql查询语句,会将该对象从数据库中查询出来:hibernate
Hibernate: select user0_.id as id0_0_, user0_.username as username0_0_, user0_.password as password0_0_, user0_.born as born0_0_ from user user0_ where user0_.id=?User [id=2, username=aaa, password=111, born=2013-10-16 00:14:24.0]
这个时候咱们可能会想,那么既然调用load方法时,并不会发出sql语句去从数据库中查出该对象,那么这个User对象究竟是个什么对象呢?代理
其实这个User对象是咱们的一个代理对象,这个代理对象仅仅保存了id这个属性:
session == (User)session.load(User., 2
咱们看到,若是咱们只打印出这个user对象的id值时,此时控制台会打印出该id值,可是一样不会发出sql语句去从数据库中去查询。这就印证了咱们的这个user对象仅仅是一个保存了id的代理对象,但若是我须要打印出user对象的其余属性值时,这个时候会不会发出sql语句呢?答案是确定的:
session = HibernateUtil.openSession(); /* * 经过load的方式加载对象时,会使用延迟加载机制,此时获得的User对象实际上是一个 * 代理对象,该代理对象里面仅仅只有id这个属性 */ User user = (User)session.load(User.class, 2); System.out.println(user.getId()); // 若是此时要获得user其余属性,则会从数据库中查询 System.out.println(user.getUsername());
此时咱们看控制台的输出:
2Hibernate: select user0_.id as id0_0_, user0_.username as username0_0_, user0_.password as password0_0_, user0_.born as born0_0_ from user user0_ where user0_.id=?aaa
相信经过上述的几个例子,你们应该很好的了解了load的这种加载对象的方式了吧。
2、get加载方式
相对于load的延迟加载方式,get就直接的多,当咱们使用session.get()方法来获得一个对象时,无论咱们使不使用这个对象,此时都会发出sql语句去从数据库中查询出来:
session = HibernateUtil.openSession(); /* * 经过get方法来加载对象时,无论使不使用该对象,都会发出sql语句,从数据库中查询 */ User user = (User)session.get(User.class, 2);
此时咱们经过get方式来获得user对象,可是咱们并无使用它,可是咱们发现控制台会输出sql的查询语句:
Hibernate: select user0_.id as id0_0_, user0_.username as username0_0_, user0_.password as password0_0_, user0_.born as born0_0_ from user user0_ where user0_.id=?
所以咱们能够看到,使用load的加载方式比get的加载方式性能要好一些,由于load加载时,获得的只是一个代理对象,当真正须要使用这个对象时再去从数据库中查询。
3、使用get和load时的一些小问题
当了解了load和get的加载机制之后,咱们此时来看看这两种方式会出现的一些小问题:
①若是使用get方式来加载对象,当咱们试图获得一个id不存在的对象时,此时会报NullPointException的异常
session = HibernateUtil.openSession(); /* * 当经过get方式试图获得一个id不存在的user对象时,此时会报NullPointException异常 */ User user = (User)session.get(User.class, 20); System.out.println(user.getUsername());
此时咱们看控制台的输出信息,会报空指针的异常:
Hibernate: select user0_.id as id0_0_, user0_.username as username0_0_, user0_.password as password0_0_, user0_.born as born0_0_ from user user0_ where user0_.id=?java.lang.NullPointerException .........
这是由于经过get方式咱们会去数据库中查询出该对象,可是这个id值不存在,因此此时user对象是null,因此就会报NullPointException的异常了。
②若是使用load方式来加载对象,当咱们试图获得一个id不存在的对象时,此时会报ObjectNotFoundException异常:
session = HibernateUtil.openSession(); /* * 当经过get方式试图获得一个id不存在的user对象时,此时会报ObjectNotFoundException异常 */ User user = (User)session.load(User.class, 20); System.out.println(user.getId()); System.out.println(user.getUsername());
咱们看看控制台的输出:
20Hibernate: select user0_.id as id0_0_, user0_.username as username0_0_, user0_.password as password0_0_, user0_.born as born0_0_ from user user0_ where user0_.id=?org.hibernate.ObjectNotFoundException: No row with the given identifier exists: [com.xiaoluo.bean.User#20]......
为何使用load的方式和get的方式来获得一个不存在的对象报的异常不一样呢??其缘由仍是由于load的延迟加载机制,使用load时,此时的user对象是一个代理对象,仅仅保存了当前的这个id值,当咱们试图获得该对象的username属性时,这个属性实际上是不存在的,因此就会报出ObjectNotFoundException这个异常了。