zoukankan      html  css  js  c++  java
  • 深入了解Hibernate的缓存使用

    Hibernate缓存

          缓存是计算机领域的概念,它介于应用程序和永久性数据存储源(如在硬盘上的文件或者数据库)之间,其作用是降低应用程序

    直接读写永久性数据存储源的频率,从而提高应用的运行性能。缓存中的数据是数据存储源中数据的复制,应用程序在运行时直接

    读写缓存中的数据,只在某些特定时刻按照缓存中的数据来同步更新数据存储源。

          缓存的物理介质通常是内存,而永久性数据存储源的物理介质通常是硬盘或磁盘,应用程序读写内存的速度显然比读写硬盘的速度

    快。如果缓存中存放的数据量非常大,也会用硬盘作为缓存的物理介质。

    Hibernate缓存一般分为3类:

    (1)一级缓存。Session缓存称为一级缓存。由于Session对象的生命周期通常对应一个数据库事务,因此它的缓存是事务范围的缓存。

    一级缓存是必需的,Session默认带有,而且不能卸载。在一级缓存中,持久化类的每个实例都具有唯一的OID。

    (2)二级缓存。SessionFactory缓存分为内置缓存和外置缓存。内置缓存是Hibernate自带的,不可拆卸,是只读缓存,用来存放映射

    元数据和预定义SQL语句。外置缓存是一个可配置的缓存插件,默认SessionFactory不会启用这个缓存插件。外置缓存中的数据是数据库

    数据的复制。SessionFactory对象的生命周期和应用程序的整个进程对应。二级缓存是可选的,可以在每个类或每个集合的粒度上配置二

    级缓存。

    (3)查询缓存。它是Hibernate为查询结果提供的,依赖于二级缓存。

    缓存的作用分为3类

    (1)事务范围。每个事务都有自己的缓存,缓存内数据不会被多个事务并发访问。例如hibernate的一级缓存,事务是不能跨多个Session的,Session内

    数据只能被当前事务访问,因此它属于事务范围的缓存。

    (2)进程范围。进程内的所有事物共享缓存,进程结束,缓存结束生命周期,例如hibernate的二级缓存,SessionFactory对象的生命周期对应应用程序的

    整个进程,因此它属于进程范围的缓存。

    (3)集群范围。缓存被一个或多个机器上的多个进程共享。Hibernate的二级缓存也可以作为集群范围的缓存。

    一级缓存:

        一级缓存的生命周期和session的生命周期一致,当前session一旦关闭,一级缓存就消失了,因此一级缓存也叫session级的缓存或事务级缓存,一级缓存只存实体对象,它不会缓存一般的对象属性(查询缓存可以),即当获得对象后,就将该对象缓存起来,如果在同一session中再去获取这个对象时,它会先判断在缓存中有没有该对象的id,如果有则直接从缓存中获取此对象,反之才去数据库中取,取的同时再将此对象作为一级缓存处理。

           
     

    内存  dept对象   内存地址:001

     deptNo

     deptName

     
       

    一级缓存

    Key:1

    Value:引用  001

     
     

    以下方法支持一级缓存:

    * get() 
        * load() 
        * iterate(查询实体对象) 
    其中 Query 和Criteria的list() 只会缓存,但不会使用缓存(除非结合查询缓存)。

     二级缓存:

    开发中的用途没有面试带来作用大。

    二级缓存是进程(N个事务)或集群范围内的缓存,可以被所有的Session共享,在多个事务之间共享

    二级缓存是可配置的插件

     配置二级缓存的步骤:

     *1.引入如下jar包。

          ehcache-1.2.3.jar  核心库

          backport-util-concurrent.jar 

          commons-logging.jar

       *2.配置Hibernate.cfg.xml开启二级缓存

       <property name="hibernate.cache.use_second_level_cache">true</property>

      *3.配置二级缓存的供应商

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

      *4.指定使用二级缓存的类

           方案一:在*.hbm.xml中配置

           在<class>元素的子元素下添加chche子节点,但该配置仅会缓存对象的简单属性,若希望缓存集合属性中的元素,必须在set元素中添加<cache>子元素

          <class name="Student" table="STUDENT">

           <cache usage="read-write"/>

    方案二:在大配置文件(hibernate.cfg.xml)中配置

    位置有限定

    Multiple annotations found at this line:

         - The content of element type "session-factory" must match "(property*,mapping*,(class-cache|collection-

          cache)*,event*,listener*)".

       - Start tag of element <session-factory>

         <class-cache   usage="read-write" class="cn.happy.entity.Student"/>

    <collection-cache usage="read-write" collection=""/>

    *5.在src下添加ehcache.xml文件,从etc获取文件即可。

    解析 :出现如下错误因为没有添加二级缓存所需jar包

    org.hibernate.HibernateException: could not instantiate RegionFactory [org.hibernate.cache.impl.bridge.RegionFactoryCacheProviderBridge]

    二级缓存分为:

           缓存算法 

           类级别缓存区

           集合级别缓存区

           更新时间戳 

           查询缓存

    query 的list()和iterate()区别 (必记)

       解析:

    1.返回的类型不一样,list返回List,iterate返回Iterator,
    2.获取数据的方式不一样,list会直接查数据库,iterate会先到数据库中把id都取出来,然后真正要遍历某个对象的时候先到缓存中找,如果找不到,以id为条件再发一条sql到数据库,这样如果缓存中没有数据,则查询数据库的次数为n+1。
    3.iterate会查询2级缓存,list 只会缓存,但不会使用缓存(除非结合查询缓存)。
    4.list中返回的List中每个对象都是原本的对象,iterate中返回的对象是代理对象

     查询缓存:

    1查询是数据库技术中最常用的操作,Hibernate为查询提供了缓存,用来提高查询速度,优化查询性能

    相同HQL语句检索结果的缓存!

    2查询缓存依赖于二级缓存

       查询缓存是针对普通属性结果集的缓存,对实体对象的结果集只缓存id(其id不是对象的真正id,可以看成是HQL或者SQL语句,它与查询的条件相关即where后的条件相关,不同的查询条件,其缓存的id也不一样)。查询缓存的生命周期,当前关联的表发生修改或是查询条件改变时,那么查询缓存生命周期结束,它不受一级缓存和二级缓存生命周期的影响,要想使用查询缓存需要手动配置如下:

    * 在hibernate.cfg.xml文件中启用查询缓存,如: 
        <property name="hibernate.cache.use_query_cache">true</property>
        * 在程序中必须手动启用查询缓存,如: 
        query.setCacheable(true); 
    其中 Query 和Criteria的list() 就可利用到查询缓存了。

    案例:

     //查询缓存
      @Test
      public void queryTest(){
          
          Session session = HibernateUtils.currentSession();
           Transaction tx=session.beginTransaction();
            List<Dept> list = session.createQuery("from Dept").setCacheable(true).list();
            System.out.println(list.get(0).getDeptName());
            
          tx.commit();
          HibernateUtils.closeSession();
          Session session2 = HibernateUtils.currentSession();
          Transaction tx2=session2.beginTransaction();
          List<Dept> list2 = session2.createQuery("from Dept").setCacheable(true).list();
           System.out.println(list.get(0).getDeptName());
          tx2.commit();
    
      }

    总结:

         不要想当然的以为缓存可以提高性能,仅仅在你能够驾驭它并且条件合适的情况下才是这样的。hibernate的二级缓存限制还是比较多的。在不了解原理的情况下乱用,可能会有1+N的问题。不当的使用还可能导致读出脏数据。 如果受不了hibernate的诸多限制,那么还是自己在应用程序的层面上做缓存吧。 

         在越高的层面上做缓存,效果就会越好。就好像尽管磁盘有缓存,数据库还是要实现自己的缓存,尽管数据库有缓存,咱们的应用程序还是要做缓存。因为底层的缓存它并不知道高层要用这些数据干什么,只能做的比较通用,而高层可以有针对性的实现缓存,所以在更高的级别上做缓存,效果也要好些吧。

          缓存是位于应用程序与物理数据源之间,用于临时存放复制数据的内存区域,目的是为了减少应用程序对物理数据源访问的次数,从而提高应用程序的运行性能. 
        Hibernate在查询数据时,首先到缓存中去查找,如果找到就直接使用,找不到的时候就会从物理数据源中检索,所以,把频繁使用的数据加载到缓存区后,就可以大大减少应用程序对物理数据源的访问,使得程序的运行性能明显的提升.

  • 相关阅读:
    Picasa生成图片幻灯片页面图文教程
    Ubuntu下缓冲器溢出攻击实验(可以看看问题分析)
    redis源码笔记 aof
    redis源码笔记 bio
    redis源码笔记 slowlog
    记录一个字符数组和字符指针的不同
    redis源码笔记 rediscli.c
    redis源码笔记 redis对过期值的处理(in redis.c)
    redis源码笔记 有关LRU cache相关的代码
    redis源码笔记 initServer
  • 原文地址:https://www.cnblogs.com/System-out-println/p/5776880.html
Copyright © 2011-2022 走看看