一级缓存中的1+N问题
所谓1+N指的就是:一条查询实体对象的ID列表的查询语句和迭代查询具体的多个实体对象的查询语句
看看下面的代码及描述:
//list()操作,hibernate会马上发出查询语句查询所有的实体对象到内存中! List persons = session.createQuery("select p from Person p").list(); for (Iterator iterator = persons.iterator(); iterator.hasNext();) { Person p = (Person) iterator.next(); System.out.println(p.getName()); }
打印结果是这样的:
Hibernate: select person_.id as id0_, person_.name as name0_, person_.address as address0_, person_.qq as qq0_, person_.groupId as groupId0_ from t_person person_
小龙哥
小马哥
神马哥
-----------------------------------------------------------------------------------------
那么iterate操作呢:
//用iterate操作查询实体对象的时候,首先会发出一条查询语句查询实体对象的ID列表 Iterator iterator = session.createQuery("select p from Person p").iterate(); for (; iterator.hasNext();) { //当迭代访问具体的某个实体对象的时候,hibernate再次发出查询语句查询此实体对象的数据! Person p = (Person) iterator.next(); System.out.println(p.getName()); }
Hibernate: select person_.id as col_0_0_ from t_person person_这个是查询id的语句
以下是查找具体对象的3条语句
Hibernate: select person_.id as id0_0_, person_.name as name0_0_, person_.address as person_0_, person_.qq as qq0_0_, contactper0_.groupId as person_0_ from t_person person_ where person_.id=?
小龙哥
Hibernate: select person_.id as id0_0_, person_.name as name0_0_, person_.address as address0_0_, person_.qq as qq0_0_, person_.groupId as groupId0_0_ from t_person person_ where person_.id=?
小马哥
Hibernate: select person_.id as id0_0_, person_.name as name0_0_, person_.address as address0_0_, person_.qq as qq0_0_, person_.groupId as groupId0_0_ from t_person person_ where person_.id=?
神马哥
现在来将他们在同一个session中查找两次看看有什么现象?sql语句就不写了
//list()操作,hibernate会马上发出查询语句查询所有的实体对象到内存中!
List persons =
session.createQuery("select p from Person p").list();
for (Iterator iterator = persons.iterator(); iterator.hasNext();) {
Person p = (Person) iterator.next();
System.out.println(p.getName());
}
//第二次,还是用list操作查询实体对象
//hibernate还会发查询语句查询实体对象
//所以,list操作会把实体对象放入一级缓存,但list操作不利用一级缓存!
persons = session.createQuery("select p from Person p")
.list();
for (Iterator iterator = persons.iterator(); iterator.hasNext();) {
Person p = (Person) iterator.next();
System.out.println(p.getName());
}
那么再看看两个iterate操作
//用iterate操作查询实体对象的时候,首先会发出一条查询语句查询实体对象的ID列表
Iterator iterator = session.createQuery("select p from Person p").iterate();
for (; iterator.hasNext();) {
//当迭代访问具体的某个实体对象的时候,hibernate再次发出查询语句查询此实体对象的数据!
Person p = (Person) iterator.next();
System.out.println(p.getName());
}
//第二次用iterate操作
//iterate操作只会再发一条查询ID列表的SQL语句,针对这些实体对象访问的时候,不再发出查询语句
//iterate操作除了能够将实体对象放入一级缓存之外,它还会利用一级缓存!!
iterator = session.createQuery("select p from Person p").iterate();
for (; iterator.hasNext();) {
//不再发出SQL查询语句!
Person p = (Person) iterator.next();
System.out.println(p.getName());
}
看到这样的结果,我们是不是可以把这两个操作结合起来提高查询性能呢?其实是可以的,再看看下面的代码
//list()操作,hibernate会马上发出查询语句查询所有的实体对象到内存中! List persons = session.createQuery("select p from Person p").list(); for (Iterator iterator = persons.iterator(); iterator.hasNext();) { Person p = (Person) iterator.next(); System.out.println(p.getName()); } //下面iterate操作,将只发出一条查询ID列表的SQL语句 Iterator iterator = session.createQuery("select p from Person p").iterate(); for (; iterator.hasNext();) { //不再发出SQL语句! Person p = (Person) iterator.next(); System.out.println(p.getName()); }
显然上面的代码分别利用list和iterate的特性,大大提高了数据查询的性能,如果不是list而是get操作,其实是一样的。
那如果查询的是一些普通结果集呢?所谓普通结果集就是某实体的某些属性,而上面查找的都是实体,看看下面的代码:
//如果查询的不是实体对象,而是一些普通的结果集,则list操作和iterate操作将没有区别 //不管是List还是iterate操作,每次查询,都会发出SQL语句!!!不再利用一级缓存!! //也就是说,一级缓存中缓存的数据只是实体 对象,而不是一般的普通结果集! List persons = session.createQuery("select p.id,p.name from Person p").list(); for (Iterator iterator = persons.iterator(); iterator.hasNext();) { Object[] p = (Object[]) iterator.next(); System.out.println(p[0]+","+p[1]); } //如果查询的不是实体对象,而是一些普通的结果集,则list操作和iterate操作将没有区别 Iterator iterator = session.createQuery("select p.id,p.name from Person p").iterate(); for (; iterator.hasNext();) { Object[] p = (Object[]) iterator.next(); System.out.println(p[0]+","+p[1]); }