18、hibernate一级缓存(hibernate笔记)

一、一级缓存(工程hibernate_cache_level_1

相关实体类:
Student.java

private int id;
private String name;
private Classes classes;

Classes.java

private int id;
private String name;
private Set students; 

配置:
Student.hbm.xml




    
        
            
        
        
        
    

Classes.hbm.xml




    
        
            
        
        
        
            
            
        
    

初始化:
InitData.java

package cn.itcast.hibernate;
import org.hibernate.Session;
public class InitData {

    public static void main(String[] args) {
            Session session = HibernateUtils.getSession();

            try {
                session.beginTransaction();

                for(int i=0; i<10; i++){
                
                    Classes classes = new Classes();
                    classes.setName("班级"+i);
                    session.save(classes);
                    
                    for(int j=0; j<10; j++){
                        Student student = new Student();
                        student.setName("班级"+i+"的学生"+j);
                        
                        //在内存中建立由student指向classes的引用
                        student.setClasses(classes);
                        session.save(student);
                    }
                }
                session.getTransaction().commit();
            } catch (Exception e) {
                e.printStackTrace();
                session.getTransaction().rollback();
            } finally{
                HibernateUtils.closeSession(session);
            }
        }   
}

测试:
CacheLevel1Test.java

package cn.itcast.hibernate;
import java.io.Serializable;
import org.hibernate.Session;
import junit.framework.TestCase;
public class CacheLevel1Test extends TestCase {
    /**
     * 在同一个session中发出两次load查询
     */
    public void testCache1() {
        Session session = null;
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            Student student = (Student)session.load(Student.class, 1);
            System.out.println("student.name=" + student.getName());
            
            //不会发出sql,因为load使用缓存
            student = (Student)session.load(Student.class, 1);
            System.out.println("student.name=" + student.getName());
            
            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
    }
}

说明:此方法中我们测试了在同一个session中发出两次load查询。可以发现在第二次使用的时候是不会发出sql语句的,这是因为load支持一级缓存。

/**
     * 在同一个session中发出两次get查询
     */
    public void testCache2() {
        Session session = null;
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            Student student = (Student)session.get(Student.class, 1);
            System.out.println("student.name=" + student.getName());
            
            //不会发出sql,因为get使用缓存
            student = (Student)session.get(Student.class, 1);
            System.out.println("student.name=" + student.getName());
            
            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
    }

说明:这里是在同一个session中使用两次get方法查询,同样和上例一样,get也是支持一级缓存的。

    /**
     * 在同一个session中发出两次iterate查询实体对象
     */
    public void testCache3() {
        Session session = null;
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            Student student = (Student)session.createQuery("from Student s where s.id=1").iterate().next();
            System.out.println("student.name=" + student.getName());
            
            //会发出查询id的sql,不会发出查询实体对象的sql,因为iterate使用缓存
            student = (Student)session.createQuery("from Student s where s.id=1").iterate().next();
            System.out.println("student.name=" + student.getName());
            
            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
    }           
    
    /**
     * 在同一个session中发出两次iterate查询普通属性
     */
    public void testCache4() {
        Session session = null;
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            String name = (String)session.createQuery("select s.name from Student s where s.id=1").iterate().next();
            System.out.println("student.name=" + name);
            
            //iterate查询普通属性,一级缓存不会缓存,所以发出sql
            //一级缓存是缓存实体对象的
            name = (String)session.createQuery("select s.name from Student s where s.id=1").iterate().next();
            System.out.println("student.name=" + name);
            
            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
    }

说明:之前我们已经演示过,在使用iterate方式查询实体对象的时候是支持缓存的,但是注意会出现N+1问题,同时这里我们也可以看到hibernate的一级缓存中缓存的是实体对象,对于普通属性是不会缓存的,从这里的第二个测试方法中可以看到会发出两次sql。

    /**
     * 开启两个session中发出load查询
     */
    public void testCache5() {
        Session session = null;
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            Student student = (Student)session.load(Student.class, 1);
            System.out.println("student.name=" + student.getName());

            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
        
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            //会发出查询语句,session间不能共享一级缓存的数据
            //因为它会伴随session的生命周期存在和消亡
            Student student = (Student)session.load(Student.class, 1);
            System.out.println("student.name=" + student.getName());

            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
        
    }       

说明:这里我们使用两次session发出load查询,主要是为了说明hibernate的一级缓存的声明周期是和session一致的,所以这里会发出两次sql语句。

    /**
     * 在同一个session中先save,在发出load查询save过的数据
     */
    public void testCache6() {
        Session session = null;
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            Student stu = new Student();
            stu.setName("王五");
            
            Serializable id = session.save(stu);
            
            //不会发出sql,因为save是使用缓存的
            Student student = (Student)session.load(Student.class, id);
            System.out.println("student.name=" + student.getName());
            
            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
    }

说明:这里我们使用save存储数据之后再使用load进行查询主要是为了说明save方法是支持一级缓存的。

    /**
     * 向数据库中批量加入1000条数据
     */
    public void testCache7() {
        Session session = null;
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            for (int i=0; i<1000; i++) {
                Student student = new Student();
                student.setName("s_" + i);
                session.save(student);
                //每20条数据就强制session将数据持久化
                //同时清除缓存,避免大量数据造成内存溢出
                if ( i % 20 == 0) {
                    session.flush();
                    session.clear();
                }
            }
            
            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
    }   

说明:hibernate的一级缓存是不能够取消的,但是我们却可以对它进行管理。比如这里我们需要批量存入1000条数据,那么我们不能一下在将所有数据都放到缓存中,那样可能造成内存溢出。所有我们选择每存20条数据就对缓存进行一次清理,使用clear方法。

最后:如果数据量特别大,可以考虑采用纯JDBC实现,如果这样还不行,可以考虑采用数据库本身的特定导入工具。一般对于大量数据的处理一般不实用Hibernate。

你可能感兴趣的:(18、hibernate一级缓存(hibernate笔记))