风雨无阻

hibernate二级缓存的实现

于Hibernate这类ORM而言,的尤重要,是持久性能提升的关键.简单来讲Hibernate就是JDBC行封装,以实现内状态的管理,OR系的映射等,但带来的就是访问效率的降低,和性能的下降,而存就是弥补这一缺点的重要方法.

    
存就是库数据在存中的临时容器,包括库数据在存中的临时,位于库与数库访问层.ORM查询数首先根据自身的存管理策略,在存中找相关数据,如发现所需的据,直接据作为结果加以利用,而避免了库调用性能的开销.而相对内存操作而言,库调用是一代价高程.

    一般
来讲ORM中的存分以下几:

        1.
务级缓存:即在前事围内存.就Hibernate来讲,务级缓存是基于Session的生命周期实现的,每Session存在一个数存,它随着Session的建而存在,着Session的毁而亡,因此也称为Session Level Cache.

        2.
级缓存:即在某个应用中或用中某个独库访问子集中的共享存,此存可由多共享(用事),存共享策略与应用的事隔离机制密切相.在Hibernate中,级缓存由SessionFactory实现,所有由一SessionFactory建的Session例共享此存,因此也称为SessionFactory Level Cache.

        3.
分布式存:即在多个应例,多JVM共享的存策略.分布式存由多个应级缓成,通种远程机制(RMI,JMS)实现个缓据同步,任何一个实例的据修改,将导致整集群状态同步.

    Hibernate
存:

        1.
存(Session Level Cache也级缓存):

        
明:

java 代

public class Test {    
  
      public void get(){    
  
            Session session = HibernateSessionFactory.getSession();    
            TUser t = (TUser)session.get("hibernate.TUser", 2);    
            System.out.println(t.getName());    
            session.close();    
            }    
  
}    
  

            
测试:在控制台打印出一SQL句:Hibernate: select tuser0_.id as id0_0_, tuser0_.name as name0_0_, tuser0_.sex as sex0_0_ from test.t_user tuser0_ where tuser0_.id=? 行了一次用.

      代
更改如下:

public class Test {    
    
      public void get(){    
  
            Session session = HibernateSessionFactory.getSession();    
            TUser t = (TUser)session.get("hibernate.TUser", 2);    
            System.out.println(t.getName());    
            TUser tt = (TUser)session.get("hibernate.TUser", 2);    
            System.out.println(tt.getName());    
            session.close();    
  
      }    
  
}    
  

       再
测试:行了查询,控制台仍然只打出一SQL句:Hibernate: select tuser0_.id as id0_0_, tuser0_.name as name0_0_, tuser0_.sex as sex0_0_ from test.t_user tuser0_ where tuser0_.id=?  是只行了一次用.

       再
更改如下:

public class Test {    
    
      public void get(){    
  
            Session session = HibernateSessionFactory.getSession();    
            TUser t = (TUser)session.get("hibernate.TUser", 2);    
            System.out.println(t.getName());    
            session.close();    
            Session session1 = HibernateSessionFactory.getSession();    
            TUser tt = (TUser)session1.get("hibernate.TUser", 2);    
            System.out.println(tt.getName());    
            session1.close();    
  
      }    
  
}    

      
继续测试:查询控制台打印两条SQL句:Hibernate: select tuser0_.id as id0_0_, tuser0_.name as name0_0_, tuser0_.sex as sex0_0_ from test.t_user tuser0_ where tuser0_.id=?
Hibernate: select tuser0_.id as id0_0_, tuser0_.name as name0_0_, tuser0_.sex as sex0_0_ from test.t_user tuser0_ where tuser0_.id=?

      
结论:Hibernate查询时总是先在存中查询,存中有所需据才查询.Hibernate存是基于Session的生命周期的,也就是存在于每Session部,它随着Session的建而存在,着Session的毁而亡,存一般由Hibernate自动维护,不需要人,然我也可以根据需要行相操作:Session.evict(Object)(指定从内除),Session.clear()(存).(如在查询间加入Session.clear()将会清存,使得一Sesion部的次相同的查询对数库进次操作).

      2.二
级缓存:(有时称为SessionFactory Level Cache)

      Hibernate
本身未提供二级缓存的品化实现(只提供了一基于HashTable的简单缓存以供调试),里我使用的是第三方件:EHcache.Hibernate的二级缓实现需要行以下配置(Hibernate3):

      首先在hibernate.cfg.xml
添加:



<property name="cache.provider_class">org.hibernate.cache.EhCacheProvider</property>  
<property name="hibernate.cache.use_query_cache">true</property>  

然后在映射文件中添加:
<cache usage="read-only"/>  

            
测试上面代:控制台出多了这样一句[ WARN] (CacheFactory.java:43) - read-only cache configured for mutable class: hibernate.TUser,二级缓用成功!!      

java 代

public class Test {    
    
      public void executeQuery(){    
      
            List list = new ArrayList();    
            Session session = HibernateSessionFactory.getSession();    
            Query query = session.createQuery("from TUser t");    
            query.setCacheable(true);//
激活查询缓存    
            list = query.list();    
            session.close();    
  
      }    
      public void get(){    
  
            Session session = HibernateSessionFactory.getSession();    
            TUser t = (TUser)session.get("hibernate.TUser", 2);    
            System.out.println(t.getName());    
            session.close();    
  
     }    
  
}    

      
测试:控制台只出一SQL句:Hibernate: select tuser0_.id as id0_, tuser0_.name as name0_, tuser0_.sex as sex0_ from test.t_user tuser0_(即Query query = session.createQuery("from TUser t")句代对应的SQL).  executeQuery()方法get()方法使用的是不同的Session!!可是executeQuery()方法get()方法只对数库进行了一次操作,就是二级缓存在起作用了.  

      
结论:Hibernate级缓存是SessionFactory存,Session共享,使用需要使用第三方的件,新版HibernateEHcache的二级缓实现.

      
存同步策略:存同步策略定了象在存中的存取规则,须为个实指定相存同步策略.Hibernate中提供了4不同的存同步策略:(暂时记个概)

      1.read-only:
.于不会发生改据可使用(对数据只能查询,其他的增删改都会报错不关是1或2缓存中.

      2.nonstrict-read-write:
如果程序对并发访问下的据同步要求不格,且据更新低,采用本存同步策略可好性能.(不能在二级缓存进行增删改都会报错)

      3.read-write:
格的读写缓存.基于时间戳判定机制,实现了"read committed"事隔离等.用于对数据同步要求的情,但不支持分布式存,实际应用中使用最多的存同步策略.(都可以比较常用的)

      4.transactional:
存,必须运行在JTA事务环境中.此存中,存的相操作被添加到事中(此似于一个内),如事,则缓冲池的一同回到事始之前的状态.实现了"Repeatable read"事隔离等,有效保据的合法性,适对关键数据的存,Hibernate存中,只有JBossCache支持事存.


create table teamEH (id varchar(32),teamname varchar(32));

create table studentEH (id varchar(32),name varchar(32),team_id varchar(32));

POJO:

 

package EHCache;

public class Student ...{

    private String id; //标识id

    private String name; //学生姓名

    private Team team;//班级

    public String getName() ...{

        return name;

    }

   

    public void setId(String id) ...{

        this.id = id;

    }

   

    public void setName(String stuName) ...{

        this.name = stuName;

    }

  

    public String getId() ...{

        return id;

    }

    public Student() ...{ //无参的构造函数

    }

   

    public Team getTeam() ...{

        return team;

    }

    public void setTeam(Team team) ...{

        this.team = team;

    }

}

package EHCache;

import java.util.HashSet;

import java.util.Set;

public class Team ...{

    private String id;

    private Set students;

    private String teamName;

    public String getId() ...{

        return id;

    }

    public void setId(String id) ...{

        this.id = id;

    }

    public String getTeamName() ...{

        return teamName;

    }

    public void setTeamName(String name) ...{

        this.teamName = name;

    }

    public Set getStudents() ...{

        return students;

    }

    public void setStudents(Set students) ...{

        this.students = students;

    }

}

 Team.hbm.xml

其中<cache>标签表示对student集合缓存,但只缓存id,如果需要缓存student实例,则需要在student.hbm.xml中的

class标签中配置<cache>

 

<?xml version="1.0" encoding="utf-8"?>

<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"

"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">

<!-- 

    Mapping file autogenerated by MyEclipse - Hibernate Tools

-->

<hibernate-mapping package="EHCache" >

    <class name="EHCache.Team" table="teamEH" lazy="false">

       <id name="id" column="id">

         <generator class="uuid.hex"></generator>

       </id>

       <property name="teamName" column="teamName"></property>

       

       <set name="students" 

            lazy="true" 

            inverse="true" 

            outer-join="false"

            batch-size="2"

            cascade="save-update"

           >

           <!-- 对students集合缓存,但只是缓存student-id如果要对整个对象缓存,

                还需要在Student.hbm.xml的class标签中加入<cache>标签 -->

         <cache usage="read-write"/>

         <key column="team_id"></key>

         <one-to-many class="EHCache.Student"/>

       </set>

      </class>

</hibernate-mapping>

 

Student.hbm.xml

 

<?xml version="1.0" encoding="utf-8"?>

<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"

"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">

<!-- 

    Mapping file autogenerated by MyEclipse - Hibernate Tools

-->

<hibernate-mapping package="EHCache" >

   

    <class name="EHCache.Student" table="studentEH" lazy="false">

       <cache usage="read-write"/>

       <id name="id" column="id" unsaved-value="null">

         <generator class="uuid.hex"></generator>

       </id>

       <property name="name" column="name"></property>

    

       <many-to-one name="team" 

                    column="team_id"

                    outer-join="true" 

                    cascade="save-update"

                    class="EHCache.Team"></many-to-one>

      </class>

</hibernate-mapping>

 

Hibernate.cfg.xml

配置hibernate.cache.provider_class以启用EHCache

<?xml version='1.0' encoding='UTF-8'?>

<!DOCTYPE hibernate-configuration PUBLIC

          "-//Hibernate/Hibernate Configuration DTD 3.0//EN"

          "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">

<!-- Generated by MyEclipse Hibernate Tools.                   -->

<hibernate-configuration>

<session-factory>

    <property name="connection.username">root</property>

    <property name="connection.url">

        jdbc:mysql://localhost:3306/schoolproject?characterEncoding=gb2312&amp;useUnicode=true

    </property>

    <property name="dialect">

        org.hibernate.dialect.MySQLDialect

    </property>

    <property name="myeclipse.connection.profile">mysql</property>

    <property name="connection.password">1234</property>

    <property name="connection.driver_class">

        com.mysql.jdbc.Driver

    </property>

    <property name="hibernate.dialect">

        org.hibernate.dialect.MySQLDialect

    </property>

    <property name="hibernate.show_sql">true</property>

    <property name="current_session_context_class">thread</property>

    <property name="hibernate.cache.provider_class">

            org.hibernate.cache.EhCacheProvider

        </property>

    <mapping resource="EHCache/Student.hbm.xml" />

    <mapping resource="EHCache/Team.hbm.xml" />

</session-factory>

</hibernate-configuration>

EHCache.xml(放在classpath下)

 

<ehcache>

 

    <diskStore path="c:\cache"/>  <!--缓存文件存放位置-->

    <defaultCache

        maxElementsInMemory="10000"

        eternal="false"

        timeToIdleSeconds="120"

        timeToLiveSeconds="120"

        overflowToDisk="true"

        />

    <cache name="EHCache.Student"

        maxElementsInMemory="500"    <!---超过500实例,就将多出的部分放置缓存文件中->

        eternal="false"

        timeToIdleSeconds="120"

        timeToLiveSeconds="120"

        overflowToDisk="true"

        /> -->

    <!-- Place configuration for your caches following -->

</ehcache>

 

测试代码(插入准备数据部分)

 

package EHCache;

import java.io.File;

import java.util.List;

import org.hibernate.Session;

import org.hibernate.SessionFactory;

import org.hibernate.Transaction;

import org.hibernate.cfg.Configuration;

public class Test ...{

    public static void main(String[] args) ...{

        String filePath=System.getProperty("user.dir")+File.separator+"src/EHCache"+File.separator+"hibernate.cfg.xml";

        File file=new File(filePath);

        SessionFactory sessionFactory=new Configuration().configure(file).buildSessionFactory();

        Session session=sessionFactory.openSession();

        Transaction tx=session.beginTransaction();

        

//        Team team=new Team();

//        team.setTeamName("team1");

//        

//        

//        for(int i=0;i<1000;i++){

//            Student stu=new Student();

//            stu.setName("tom"+i);

//            stu.setTeam(team);

//            session.save(stu);

//        }

//        tx.commit();

//        

    }

}

 

测试成功后,运行以下代码

 

package EHCache;

import java.io.File;

import java.util.List;

import org.hibernate.Session;

import org.hibernate.SessionFactory;

import org.hibernate.Transaction;

import org.hibernate.cfg.Configuration;

public class Test ...{

    public static void main(String[] args) ...{

        String filePath=System.getProperty("user.dir")+File.separator+"src/EHCache"+File.separator+"hibernate.cfg.xml";

        File file=new File(filePath);

        SessionFactory sessionFactory=new Configuration().configure(file).buildSessionFactory();

        Session session=sessionFactory.openSession();

        Transaction tx=session.beginTransaction();

        

    

        //模拟多用户访问数据

        Session session1=sessionFactory.openSession();

        Transaction tx1=session1.beginTransaction();

        List list=session1.createQuery("from Student").list();

        for(int i=0;i<list.size();i++)...{

            Student stu=(Student)list.get(i);

            System.out.println(stu.getName());

        }

        tx1.commit();

        session1.close();    

    

        Session session2=sessionFactory.openSession();

        Transaction tx2=session2.beginTransaction();

            //这个uuid从刚才插入的数据中复制一个student的id

        Student stu=(Student)session2.get(Student.class, "4028818316d184820116d184900e0001");

        System.out.println(stu.getName());

        tx2.commit();

        session2.close();

    }

}

 

结果如下:

log4j:WARN No appenders could be found for logger (org.hibernate.cfg.Environment).

log4j:WARN Please initialize the log4j system properly.

Hibernate: select student0_.id as id0_, student0_.name as name0_, student0_.team_id as team3_0_ from studentEH student0_

Hibernate: select team0_.id as id1_0_, team0_.teamName as teamName1_0_ from teamEH team0_ where team0_.id=?

tom0

tom1

tom2

tom3

tom4

tom5

tom6

tom7

tom8

tom9

tom10

........................................

tom974

tom975

tom976

tom977

tom978

tom998

tom999

Hibernate: select team0_.id as id1_0_, team0_.teamName as teamName1_0_ from teamEH team0_ where team0_.id=?

tom0

 

可以看到,第二次查询,已经不再访问数据库了,而且,查看c:\cache文件夹,也可以看到,数据已经缓存成功了

posted on 2008-04-16 18:17 秋枫故事 阅读(1759) 评论(1)  编辑  收藏

评论

# re: hibernate二级缓存的实现 2009-09-15 17:56 jack.lxh@gmail.com

您好文章中的说明 的hiberate 一级缓存与二级缓存的交互,我之前就知道,但是如果在一个web应用中 这个进程级别与 application 范围有什么区别  回复  更多评论   


只有注册用户登录后才能发表评论。


网站导航:
 
<2008年4月>
303112345
6789101112
13141516171819
20212223242526
27282930123
45678910

导航

统计

常用链接

留言簿(2)

随笔分类

随笔档案

新闻档案

搜索

最新评论

阅读排行榜

评论排行榜