hibernate缓存机制分析.pdf
文本预览下载声明
hibernate 缓存机制分析
一、N+1 问题
首先我们来探讨一下 N+1 的问题,我们先通过一个例子来看一下,什么是 N+1 问题:
list()获得对象:
/**
* 此时会发出一条 sql ,将30 个学生全部查询出来
*/
ListStudent ls = (ListStudent)session.createQuery(from Student)
.setFirstResult(0).setMaxResults(30).list();
IteratorStudent stus = ls.iterator();
for(;stus.hasNext();)
{
Student stu = (Student)stus.next();
System.out.println(stu.getName());
}
如果通过 list()方法来获得对象,毫无疑问,hibernate 会发出一条 sql 语句,将所有的对象查询出来,这点相信大家都能理
解
Hibernate: select student0_.id as id2_, student0_.name as name2_, student0_.rid as rid2_, student0_.sex as sex2_
from t_student student0_ limit ?
那么,我们再来看看 iterator()这种情况
iterator()获得对象
/**
* 如果使用 iterator 方法返回列表,对于 hibernate 而言,它仅仅只是发出取 id 列表的 sql
* 在查询相应的具体的某个学生信息时,会发出相应的SQL 去取学生信息
* 这就是典型的 N+1 问题
* 存在 iterator 的原因是,有可能会在一个 session 中查询两次数据,如果使用 list 每一次都会把所有的对象
查询上来
* 而是要 iterator 仅仅只会查询 id ,此时所有的对象已经存储在一级缓存(session 的缓存)中,可以直接获取
*/
IteratorStudent stus = (IteratorStudent)session.createQuery(from Student)
.setFirstResult(0).setMaxResults(30).iterate();
for(;stus.hasNext();)
{
Student stu = (Student)stus.next();
1 / 17
System.out.println(stu.getName());
}
在执行完上述的测试用例后,我们来看看控制台的输出,看会发出多少条 sql 语句:
Hibernate: select student0_.id as col_0_0_ from t_student student0_ limit ?
Hibernate: select student0_.id as id2_0_, student0_.name as name2_0_, student0_.rid as rid2_0_, student0_.sex as
sex2_0_ from t_student student0_ where student0_.id=?
Hibernate: select student0_.id as id2_0_, student0_.name
显示全部