hibernate05--list和iterator
package cn.bdqn.test; import java.util.Iterator; import java.util.List; import org.hibernate.Query; import org.hibernate.Session; import org.hibernate.SessionFactory; import org.hibernate.Transaction; import org.hibernate.cfg.Configuration; import org.junit.After; import org.junit.Before; import org.junit.Test; import cn.bdqn.bean.Student; import cn.bdqn.util.HibernateSessionUtil; public class StudentTest { Session session=null; Transaction transaction=null; @Before public void before(){ //getCurrentSession 必须在事务下运行 session=HibernateSessionUtil.getCurrentSession(); transaction=session.beginTransaction(); //开启事务 } /** * HQL: hibernate查询语言! * * 执行hql的步骤: * 01.获取session对象 * 02.编写hql语句 使用面向对象的思想! hql中只有类 和属性 !不存在 表和字段 * 03.通过session.createQuery(String hql) 创建Query对象 * 05.执行对应的查询 */ /** * list查询所有: * 01.会立即产生一条select语句! * select查询出来的所有数据,都会被session管理!保存在缓存中! * 02.清空或者不清空session缓存中的数据 * 03.再次执行查询的时候 都会执行一条select语句! */ @Test public void testList(){ //Student 必须大写 因为是 类名 String hql="from Student"; //创建Query对象 Query query = session.createQuery(hql); //执行对应的查询 System.out.println("*************"); List<Student> list = query.list(); System.out.println("*************"); for (Student student : list) { System.out.println(student); } //清空缓存 //session.clear(); //再次执行对应的查询 list = query.list(); for (Student student : list) { System.out.println(student); } } /** * Iterator:查询所有 * * 测试环境:数据库中有5条数据 * * 产生的结果: * 01.6条select语句 * 02.第一条 是查询数据库表中所有的id,这条语句是query.iterate()产生的! * 03.其他的5条select语句 都是根据id进行查询!都是在.next()产生的! */ @Test public void testIterator(){ String hql="from Student"; Query query = session.createQuery(hql); System.out.println("*************"); Iterator<Student> iterate = query.iterate(); System.out.println("*************"); while (iterate.hasNext()) { System.out.println("*************"); Student stu = iterate.next(); System.out.println("*************"); System.out.println(stu); } } /** * 01.iterate在有缓存的情况下,如果缓存中有查询的所有数据!只会执行一条sql语句! * 这条sql就是查询所有的id! * 02.如果缓存中有2条数据! id =1 id=2 * 我们查询了所有的5条数据! * 这时候会产生多少条sql? 3+1 */ @Test public void testIterator2(){ String hql="from Student"; Query query = session.createQuery(hql); Iterator<Student> iterate = query.iterate(); while (iterate.hasNext()) { Student stu = iterate.next(); System.out.println(stu); } System.out.println("********************"); //再次查询 没有清空缓存 iterate = query.iterate(); while (iterate.hasNext()) { Student stu = iterate.next(); System.out.println(stu); } } /** * 测试环境: * 缓存中有两条数据 * 结果: * 01.get肯定产生sql * 02.iterate遍历的时候 先去缓存中获取已经存在的数据! 就会减少2次查询! */ @Test public void testIterator21(){ //获取id为1的student对象 Student student1= (Student) session.get(Student.class, 1); // 产生1条 Student student2 = (Student) session.get(Student.class, 2); // 产生1条 System.out.println("*************************"); String hql="from Student"; Query query = session.createQuery(hql); Iterator<Student> iterate = query.iterate(); // 产生1条 while (iterate.hasNext()) { Student stu = iterate.next();// 产生4条 System.out.println(stu); } } /** *iterate在没有缓存的情况下 会执行N+1条数据! *N:指的是数据数量! *1:查询所有的ID! */ @Test public void testIterator3(){ String hql="from Student"; Query query = session.createQuery(hql); Iterator<Student> iterate = query.iterate(); while (iterate.hasNext()) { Student stu = iterate.next(); System.out.println(stu); } System.out.println("********************"); //再次查询 清空缓存 session.clear(); iterate = query.iterate(); while (iterate.hasNext()) { Student stu = iterate.next(); System.out.println(stu); } }
}
/** Query接口中的list()和iterate()都可以执行查询操作,
而iterate()能够利用延迟加载和缓存的机制提高查询性能!iterate()查询时,
仅查询ID字段以节省资源。需要使用数据时,再根据ID字段到缓存中检索匹配的实例!
如果存在就直接使用!只有当缓存中没有需要的数据时,iterate()才会执行select语句
!根据ID字段到数据库中查询!iterate()更适用于查询对象开启二级缓存的情况! */