hibernate.cache.use_structured_entries

Hibernate文档上介绍,该属性是用于把对象以一种更易读的方式放到二级缓存中,这样,在对二级缓存进行监控的时候就更容易理解放到二级缓存中对象的内容。只需要简单的设置

  1.   hibernate.cache.use_structured_entries true

即可。

那这个属性到底有什么作用呢?先简单看一下效果:准备数据代码和映射文件在前面一篇帖子中有:

http://blog.csdn.net/stefwu/article/details/10349407

下面是测试代码:

  1.   @Test
  2.    public void testStatistics4() {
  3.    Session session = HibernateUtil.getInstance().getSession();
  4.    session.get(Department.class, 1l);
  5.   
  6.    Statistics st = HibernateUtil.getInstance().getSf().getStatistics();
  7.    //得到二级缓存区域统计信息
  8.    SecondLevelCacheStatistics scs = st
  9.    .getSecondLevelCacheStatistics("itcast.DEPT");
  10.    //得到该二级缓存区域中缓存的实体对象,是一个Map,遍历这个map
  11.    for(Object o:scs.getEntries().entrySet()){
  12.    AbstractReadWriteEhcacheItemReader.readItem(((Entry)o).getValue());
  13.    }
  14.    session.close();
  15.    }

该代码中有一些东西先解释一下,首先,通过SecondLevelCacheStatistics的getEntries()方法得到存放在该二级缓存区域中的实体集,接着遍历这个实体集,每一个Entry的value就是存放的实体,但是,该实体是一个org.hibernate.cache.ehcache.internal.strategy.AbstractReadWriteEhcacheAccessStrategy.Item或者org.hibernate.cache.ehcache.internal.strategy.AbstractReadWriteEhcacheAccessStrategy.Lock(锁定缓存对象,准备操作的时候)对象。其中,Item对象是用于包装真正缓存对象的东西,在其上有一个getValue()方法,该方法返回的就是保存的结构化(注意这个结构化,就是由hibernate.cache.use_structured_entries控制)的缓存对象;Lock对象是用于在锁定一个对象(只会锁定其ID,即Class#id)时生成的临时的锁对象,所以,真正我们需要观察的其实是Item对象。但是,这两个对象都是protected,所以都无法直接访问,所以,我们得写一个类专门用于访问这两个对象:

  1.   package org.hibernate.cache.ehcache.internal.strategy;
  2.   
  3.   import org.hibernate.cache.ehcache.internal.strategy.AbstractReadWriteEhcacheAccessStrategy.Item;
  4.   import org.hibernate.cache.ehcache.internal.strategy.AbstractReadWriteEhcacheAccessStrategy.Lock;
  5.   
  6.   
  7.   public class AbstractReadWriteEhcacheItemReader {
  8.   
  9.    public static void readItem(Object item){
  10.    if(item instanceof Item){
  11.    System.out.println(((Item)item).getValue());
  12.    }else if(item instanceof Lock){
  13.    System.out.println(item);
  14.    }
  15.    }
  16.   }

注意,包一定是org.hibernate.cache.ehcache.internal.strategy。

那么我们运行上面的测试代码就会看到输出:

  1.   {emps=1, _version=null, _lazyPropertiesUnfetched=false, name=d1, _subclass=cd.itcast.hibernate.day2.many2one.Department}

很明显,hibernate将缓存的对象变成了一个更容易理解的Map对象,在对象上有三个额外的列:_version,_subclass和_lazyPropertiesUnfetched;其中_version代表该对象的版本;_subclass代表缓存对象的真实类型(因为在从缓存把这个对象拿出去的时候,还需要靠这个标记来还原对象类型);_lazyPropertiesUnfetched代表该对象上是否还有延迟加载的属性没有获取;

而假设我们把hibernate配置还原:

  1.   hibernate.cache.use_structured_entries false

再次运行测试,控制台输出:

  1.   CacheEntry(cd.itcast.hibernate.day2.many2one.Department)[d1,1]

可以看到,如果没有格式化,那么hibernate直接保存到缓存中的是一个CacheEntry对象。

了解了对象是怎么存放在二级缓存中的,我们来看看更深一点点的东西,到底hibernate.cache.use_structured_entries这个配置控制了什么?

其实这个问题只需要简单的DEBUG一下hibernate,就能很清楚的看到整个过程,简单叙述如下:

1,在hibernate中,缓存的读取是靠EntityRegionAccessStrategy这个接口控制的,简单说,这个类控制了二级缓存的操作策略,怎么添加数据,怎么得到数据,怎么修改数据,怎么删除数据;而数据以什么样的格式存放到二级缓存中,又怎么读取成对象,这个过程由CacheEntryStructure这个接口控制;如果hibernate.cache.use_structured_entries设置为false(默认),则hibernate选择UnstructuredCacheEntry来作为存储结构,如果设置为true,则hibernate选择StructuredCacheEntry来作为存储结构。

2,在CacheEntryStructure中有两个非常重要的方法:

  1.   public interface CacheEntryStructure {
  2.    //怎么把CacheEntry对象变成要保存到二级缓存中的数据结构
  3.    public Object structure(Object item);
  4.   //怎么把二级缓存中得到的数据结构重新变成CacheEntry对象
  5.    public Object destructure(Object map, SessionFactoryImplementor factory);
  6.   }

这两个方法很好理解,简单来说就是CacheEntry对象到二级缓存中数据结构的Mapping方法。

3,在把对象存放到二级缓存中的时候,先通过CacheEntryStructure把要存放的value变成对应的存放结构:

  1.   cacheEntry = persister.getCacheEntryStructure().structure(ce);

然后调用RegionAccessStrategy上的putFromLoad(Object key,Object value,long txTimestamp,Object version)方法,把格式化之后的数据存放:

  1.   region.put( key, new Item( value, version, region.nextTimestamp() ) );

4,在从二级缓存中得到对象的时候,首先通过RegionAccessStrategy上的get(Object key,long txTimestamp)得到二级缓存中存放的对应的数据:

  1.   Object ce = persister.getCacheAccessStrategy().get( ck, source.getTimestamp() );

然后再使用CacheEntryStructure的destructure方法把对象重新变成CacheEntry给hibernate:

  1.   CacheEntry entry = (CacheEntry) persister.getCacheEntryStructure().destructure( ce, factory );

如果是设置hibernate.cache.use_structured_entries=true的时候,在读取的时候都会有个格式的转换过程,而hibernate.cache.use_structured_entries=false的时候,直接存放的就是CacheEntry,所以,建议在生产环境中还是不要设置这个属性,在对二级缓存进行调试的时候,再配置这个属性。

从配置文件入手去阅读hibernate的源代码,可以看到很多有趣的东西,而且代码也很集中。大家不妨试试。

Hibernate 配置详解(9)的更多相关文章

  1. Hibernate 配置详解(2)

    6) hibernate.session_factory_name: 配置一个JNDI名称,通过Configuration对象创建的SessionFactory会绑定到JNDI下该名称中.一般名字格式 ...

  2. Hibernate 配置详解(5)

    9) hibernate.batch_fetch_style: 该配置是hibernate4.2.0新添加的,使用这个设置可以配置hibernate在做batch-fetch的时候,生成SQL的策略. ...

  3. Hibernate 配置详解(8)

    hibernate.generate_statistics 这个配置大家应该都很熟悉,用于开启Hibernate统计信息,便于对Hibernate相关性能调试提供数据依据.在开发过程当中,可以把这个选 ...

  4. Hibernate 配置详解(12) 补充

    hibernate.hbm2ddl.import_files_sql_extractor 这个配置项用于补充这篇文章: http://blog.csdn.net/stefwu/article/deta ...

  5. Hibernate 配置详解(12) 其实我也不想用这么土的名字

    hibernate.hbm2ddl.import_files 这个配置用于在hibernate根据映射文件执行DDL之前,如果我们自己设置了要事先运行的SQL文件,hibernate就会先执行这些SQ ...

  6. Hibernate 配置详解(7)

    hibernate.order_updates: Hibernate文档中提到,该配置用于在刷新一级缓存,提交UPDATE的时候,按照每类对象的主键顺序排序后再提交,可以在高并发情况下减少事务死锁的可 ...

  7. Hibernate 配置详解(11)

    hibernate.session_factory_name_is_jndi 配置hibernate.cfg.xml中SessionFactory的name属性是否作为JNDI名称绑定.默认是true ...

  8. hibernate二级缓存ehcache hibernate配置详解

    <!-----------------hibernate二级缓存ehcache------------------------->hibernate配置 <prop key=&quo ...

  9. Hibernate配置详解

    <!--标准的XML文件的起始行,version='1.0'表明XML的版本,encoding='gb2312'表明XML文件的编码方式--> <?xml version='1.0' ...

随机推荐

  1. 每日一dp(2)——龟兔赛跑(hdu 2059)

    比較经典的动态规划的题目了 一般动态规划的想法都是先推断是否有最优子结构,无后效性.接着从状态转移入手,尽量细分状态(即给定N得到N+1),完了再递推计算 难点:转移方程,其一般也难在怎样描写叙述一个 ...

  2. VIM 中 查看{}是否闭合,按%跳转到下个闭合

    VIM 中 查看{}是否闭合,按%跳转到下个闭合

  3. IE下图片切换的时候,图片总是切换不成功---根本问题是IE缓存图片

    作为WEB设计者,为了在网页展示上加强用户体验,经常会利用图象载入显示状态方法,这自然需要Image对象的onload事件. 在firefox浏览器下完成开发后,可是在IE浏览器中进行调试总不能被调用 ...

  4. 开启MSSQLServer跨服务器查询功能

    首先在MSSQL客户端中进行如下图文操作配置 其次使用脚本进行操作配置 ---开启SQLServer 跨服务器查询功能 exec sp_configure 'show advanced options ...

  5. JavaWeb核心编程之(四.1)JSP

    JSP简介: JSP是简化Servlet编写的一种技术, 它将Java代码和HTML语句混合在一个文件中编写, 只对网页中药动态产生的内容采用Java代码来编写, 而对固定不变的静态内容采用普通的静态 ...

  6. grep 和 perl多个条件匹配

    grep和perl多个条件匹配使用‘|’作为分割符号 grep -E 'abc|def' perl if(/abc|def/)

  7. 如果ie6跳转

    try { (function(e) { /i.test(navigator.userAgent)) { window.location = "jump.html"; return ...

  8. 关于使用HttpSessionBindingListener获取在线用户数,同一用户登陆一次

    原创地址:http://blog.csdn.net/jiaoxueli/article/details/2226134 考虑到项目中统计在线用户数量和同一用户只能登陆一次的需求,查询联系 HttpSe ...

  9. Javascript 链式运动框架——逐行分析代码,让你轻松了解运动的原理

    所谓链式运动,就是一环扣一环.我们的很多运动实际上来说指的就是分阶段的,第一个阶段动完,下个阶段开始动. 这个链式运动框架就是用来处理这些问题的. 我们先来看下之前的运动框架,以下是Javascrip ...

  10. iOS中不透明度的查看

    模拟器工具条 Debug-->Color Blended Layers 即中文显示下 调试 -->颜色混合层 绿色代表不透明部分,红色代表透明部分,红色越多对性能影响越大