hibernate二级缓存的实现

存就是库数据在存中的临时容器,包括库数据在存中的临时,位于库与数库访问层.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文件夹,也可以看到,数据已经缓存成功了

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值