问题描述

缓存在提升应用性能,提高访问效率上都是至关重要的一步。ehcache也是广为使用的缓存之一。但是如果将一个可变的对象(如普通的POJO/List/Map等)存入缓存中,会导致怎样潜在的问题。下面来看一个例子

    //首先创建一个简单的POJO
public class Student {
private String name;
public Student(String name){
this.setName(name);
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}

之后创建一个POJO并使用ehcache来进行缓存

    @Test
public void copyOnRead(){
CacheManager cache=CacheManager.create("cache.xml"); //读取cache配置文件,如果不配置ehcache有默认的failsafe配置
Ehcache copyCache=cache.addCacheIfAbsent("copyCache"); //读取名为copyCache的cache,如果没有则创建
Student stu=new Student("张三"); //创建一个简单的POJO
Element el=new Element(1,stu); //将对象放入Element对象
copyCache.put(el); //存入缓存
stu.setName("李四"); //之后改变POJO的内容
Student cachedStu=(Student) copyCache.get(1).getObjectValue();//从缓存中取出对象
System.out.println("cache中取出的对象的名字:"+cachedStu.getName());
System.out.println("cache中取出的对象和源对象相等:"+(copyCache.get(1)==el));
System.out.println("两次cache中取出的对象相等"+(copyCache.get(1)==copyCache.get(1)));
}

以上代码的输出结果是

    cache中取出的对象的名字:李四
cache中取出的对象和源对象相等:true
两次cache中取出的对象相等true

可以看到缓存中的对象在外部对象被修改时也被修改了(这里只考虑内存缓存,不考虑序列化到磁盘)。

原因分析

这是由于ehcache中存放的Element对象是直接保存了原对象的引用,所以引用的内容被修改之后cache内部的值也会被修改。

解决办法

首先这种现象并非一定是有问题的,有的应用场景就是希望保存对象的引用,但是这会导致部分误操作,如果不小心无意修改了对象的值就会导致数据不准确。

知道了原因那么也就好解决问题了:

使用immutable对象

将POJO设计成immutable的,如果需要修改就新建一个新的对象,这个也是fp所推崇的方案。

使用ehcache的copyStrategy

如果是遗留项目的话上面的方法可能就不适合了,好在在ehcache中已经实现了接口可以拦截read、write等操作,并对存入、读取的Element对象进行修改

    public class MyCopyStrategy implements ReadWriteCopyStrategy<Element> {
//当write缓存的时候会调用这个方法
public Element copyForWrite(Element value, ClassLoader loader) {
Student temp=(Student) value.getObjectValue();
return new Element(value.getObjectKey(),new Student(temp.getName()+"_w"));
}
//当read缓存的时候会调用这个方法
public Element copyForRead(Element storedValue, ClassLoader loader) {
Student temp=(Student) storedValue.getObjectValue();
return new Element(storedValue.getObjectKey(),new Student(temp.getName()+"_r"));
}
}

如上只要实现了ReadWriteCopyStrategy就可以修改存入、读取的对象。这里在读取时我再名字后加了一个r,在存入时名字后加了一个w

还需要注入到cache中

    //配置文件
<cache name="copyCache" maxEntriesLocalHeap="1" eternal="false"
timeToIdleSeconds="1" timeToLiveSeconds="1" copyOnRead="true" copyOnWrite="true">
<copyStrategy class="echach2.MyCopyStrategy" />
</cache>

使用copyStrategy将strategy配置到cache中,再运行上面的代码

    张三_w_r
cache中取出的对象和源对象相等:false
两次cache中取出的对象相等false

这里可以看到存在的cache中的对象和原对象并不一样,两次cache取出来的对象也不同。

小结

ehcache的copyStratgy可以有效防止引用对象在外部被误改,但是这每次都复制一个对象,对性能还是有一定的影响的,所以可能的话还是尽量使用immutable的对象。

ehcache2拾遗之copyOnRead,copyOnWrite的更多相关文章

  1. ehcache2拾遗之write和load

    问题描述 在cache系统中writeThrough和writeBehind是两个常用的模式. writeThrough是指,当用户更新缓存时,自动将值写入到数据源. writeBehind是指,在用 ...

  2. ehcache2拾遗之cache持久化

    问题描述 应用在使用过程中会需要重启等,但是如果ehcache随着应用一起重启,那么刚重启的时候就会出现大量的miss,需要一定的访问量来重建缓存,如果缓存能够持久化,重启之后可以复用将会有助于缓解重 ...

  3. Ehcache(2.9.x) - API Developer Guide, Transaction Support

    About Transaction Support Transactions are supported in versions of Ehcache 2.0 and higher. The 2.3. ...

  4. EhCache 配置信息

    How to Size Caches 官方文档:http://ehcache.org/documentation/configuration/cache-size [maxEntriesLocalHe ...

  5. Terracotta

    Terracotta 3.2.1简介 (一) 博客分类: 企业应用面临的问题 Java&Socket 开源组件的应用 hibernatejava集群服务器EhcacheQuartzTerrac ...

  6. Redis命令拾遗二(散列类型)

    本文版权归博客园和作者吴双共同所有,欢迎转载,转载和爬虫请注明原文地址 :博客园蜗牛NoSql系列地址  http://www.cnblogs.com/tdws/tag/NoSql/ Redis命令拾 ...

  7. 基础拾遗------redis详解

    基础拾遗 基础拾遗------特性详解 基础拾遗------webservice详解 基础拾遗------redis详解 基础拾遗------反射详解 基础拾遗------委托详解 基础拾遗----- ...

  8. unixLike命令拾遗

    针对在日常工作过程中,发现的学习的漏洞和忘记的知识,进行拾遗. 编辑命令 一.vim操作 1.进入编辑模式 在光标移到将要编辑处,点击i,进入编辑模式 2.退出编辑模式 按esc或者crtl+c退出编 ...

  9. 基础拾遗------webservice详解

    基础拾遗 基础拾遗------特性详解 基础拾遗------webservice详解 基础拾遗------redis详解 基础拾遗------反射详解 基础拾遗------委托详解 基础拾遗----- ...

随机推荐

  1. phpStudy 的Apache虚拟主机配置

    放弃了wamp,朋友介绍了phpstudy,不错的一款软件,关键是能自由切换php版本.相关的阿帕奇虚拟主机配置参考:http://www.th7.cn/system/win/201506/10846 ...

  2. linux下svn常用命令

    (如果是第一次提交文件,很可能会出现“svn:'.'不是工作副本”,即当前目录不是工作副本,这个时候需要用到import: eg:svn import . url) 1.将文件checkout到本地目 ...

  3. lua 位运算

    bit = {data32={}} , do bit.data32[i] = ^(-i) end function bit:d2b( arg ) local num = tonumber( arg ) ...

  4. Python垃圾回收机制

    引用计数Python默认的垃圾收集机制是“引用计数”,每个对象维护了一个ob_ref字段.它的优点是机制简单,当新的引用指向该对象时,引用计数 引用计数 Python默认的垃圾收集机制是“引用计数”, ...

  5. [.Net] 通过反射,给Enum加备注

    今天和大家分享一个给Enum加备注的技巧,话不多说,先上一段代码: namespace TestReflector.Model.Entities { public class UserInfo { p ...

  6. 学习django之构建Web是Meta嵌套类的几处使用

    Django中meta嵌套类的使用 1.模型中使用嵌套类 在定义抽象模型时如: class Meta : abstract=true 用来指明你创建的模型是一个抽象基础类的模型继承. 2.在一个对象对 ...

  7. NYOJ 737 石子合并(一)

    分析: 本题为区间型动态规划,dp[i][j] 表示从第 i 堆合并到第 j 堆的最小代价, sum[i][i] 表示第 i 堆到第 j 堆的石子总和,则动态转移方程: dp[i][j] = min( ...

  8. idea中maven报错:无效的目标发行版: 1.8

    1.project.pom中修改版本 <maven.compiler.source>1.7</maven.compiler.source><maven.compiler. ...

  9. select 取的是session里面的值时

    原来是写了一个select标签,然后用js循环取出来,发现问题是本来嵌在页面右边的页面整个弹出来, 后来改成html:optionsCollection就好了: 效果图:

  10. CAS 4.0.0RC 配置MD5验证功能

    配置内容同一样,只是增加一些配置. 因为cas已经默认就支持MD5加密验证,所以只是修改一下配置就可以了. <bean id="primaryAuthenticationHandler ...