在Hibernate框架中,最常用到的加载方式就非Get和Load莫属了,然而Get和Load在加载方式上边还有很多的不同,下面让我们来分析一下他们的不同之处。
区别
从返回的结果上来看,get、load在检索到数据的时候,会返回对象(代理对象或实体对象),但他们的检索方式不同,这个后续在细说。get检索不到数据的时候,会返回null;而load检索不到数据会返回org.hibernate.ObjectNotFoundException。下面通过两个例子来分析一下get与load的工作原理。
工作原理
Get加载(数据库中存在检索的数据),这里只分析测试类,实体类就不再讨论了。
/**
* 查询数据库中有数据的字段
*/
public void testGetNotNull() {
Session session = null;
try {
session = HibernateUtils.getSession();
session.beginTransaction();
// 数据库中存在id为2的数据
User user = (User)session.get(User.class, 2);
System.out.println("---------Get加载--------");
System.out.println(user);
System.out.println(user.getId() + "---->>" + user.getName());
System.out.println("------------------------");
session.getTransaction().commit();
} catch (Exception e) {
e.printStackTrace();
session.getTransaction().rollback();
} finally {
HibernateUtils.closeSession(session);
}
}
Get加载(数据库中不存在检索的数据)
/**
* 查询数据库中不存在的数据
*/
public void testGetWithNull() {
Session session = null;
try {
session = HibernateUtils.getSession();
session.beginTransaction();
// 数据库中不存在id为0的数据
User user = (User)session.get(User.class, 0);
System.out.println("---------Get加载--------");
System.out.println(user);
//System.out.println(user.getId() + "---->>" + user.getName());
System.out.println("------------------------");
session.getTransaction().commit();
} catch (Exception e) {
e.printStackTrace();
session.getTransaction().rollback();
} finally {
HibernateUtils.closeSession(session);
}
}
Load加载(数据库中存在检索的数据)
/**
* 数据库中存在查询的数据
*/
public void testLoadNotNull() {
Session session = null;
try {
session = HibernateUtils.getSession();
session.beginTransaction();
// 数据库中存在id为2的数据
User user = (User)session.load(User.class, 2);
System.out.println("----------Load加载-----------");
System.out.println(user);
System.out.println(user.getId() + "---->>" + user.getName());
System.out.println("-----------------------------");
session.getTransaction().commit();
} catch (Exception e) {
e.printStackTrace();
session.getTransaction().rollback();
} finally {
HibernateUtils.closeSession(session);
}
}
Hibernate中,load方法是支持延迟加载的,在执行到User user = (User)session.load(User.class, 2);这条语句时,load方法首先会查找session的persistent Context中是否存在缓存,如果存在则直接返回;如果没有缓存,则load方法会判断是否是lazy加载,如果不是lazy方式,则直接访问数据库,查询Id为2的数据,由于数据库中存在Id为2的数据,因此get方法会返回实体对象。
如果load方法判断的是lazy方式加载,那么就需要建立代理对象,对象的initialized属性为false,target属性为null,在访问获得的代理对象的属性时,会访问数据库,找到记录后就会把该记录的对象复制到代理对象的target上,并将initialized属性设为true。在这个例子中,load方法采用的是lazy方式,因此在执行load方法时,Hibernate不会发出sql语句,因为此时Hibernate建立了代理对象,但并没有访问代理对象的属性。当执行System.out.println(user);时,Hibernate才会初次发出sql语句,去查询数据库。如下图:
Load加载(数据库中不存在检索的数据)
/**
* 数据库中不存在查询的数据
*/
public void testLoadWithNull() {
Session session = null;
try {
session = HibernateUtils.getSession();
session.beginTransaction();
// 数据库中不存在id为0的数据
User user = (User)session.load(User.class, 0);
System.out.println("----------Load加载-----------");
System.out.println(user);
//System.out.println(user.getId() + "---->>" + user.getName());
System.out.println("-----------------------------");
session.getTransaction().commit();
} catch (Exception e) {
e.printStackTrace();
session.getTransaction().rollback();
} finally {
HibernateUtils.closeSession(session);
}
}
而执行到System.out.println(user);时,Hibernate会发出sql语句,访问数据库查询Id为0的数据,由于数据库中不存在Id为0的数据,因此load方法会返回org.hibernate.ObjectNotFoundException,直接抛出异常。如下图:
结束语
Hibernate中,对于get和load方法的根本区别,就在于Hibernate对于load方法认为数据库中一定存在要检索的数据,可以放心的使用代理来延迟加载,如果在使用过程中发现了问题,那么只能抛出异常;而对于get方法,Hibernate一定要获取到真实的数据才会返回对象,否则返回null。
参考资料:
后勤仓库的Hibernate get和load区别