首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 其他教程 > 开源软件 >

运用EhCache和Spring AOP实现计算结果缓存

2012-09-24 
使用EhCache和Spring AOP实现计算结果缓存一.Ehcache简介??? EhCache是一个纯Java的进程内缓存框架,具有如

使用EhCache和Spring AOP实现计算结果缓存

一.Ehcache简介
??? EhCache是一个纯Java的进程内缓存框架,具有如下特点:
??? 1. 快速简单,非常容易和应用集成。
??? 2.支持多种缓存策略 。
??? 3. 缓存数据有两级:内存和磁盘,因此无需担心容量问题 。
??? 4. 缓存数据会在虚拟机重启的过程中写入磁盘 。
??? 5. 可以通过RMI、可插入API等方式进行分布式缓存。
??? 6. 具有缓存和缓存管理器的侦听接口 。
??? 7. 支持多缓存管理器实例,以及一个实例的多个缓存区域 等特点。

二.Ehcache配置的相关参数
??? Ehcache的配置很灵活,官方提供的配置方式有好几种,你可以通过声明配置、在xml中配置、在程序里配置或者调用构造方法时传入不同的参数。下面以最常用的XML配置为例说下配置的相关参数的意义,ehcache.xml是最常见的一个文件,ehcache一般会通过CacheManager从classpath加载该文件完成Cache的实例化。
???
??? 1.ehcache.xml中的配置信息
??? ??? ehcache.xml片段:

      <ehcache>            <diskStore path="java.io.tmpdir"/>            <defaultCache                name="name"                    maxElementsInMemory="10000"                    eternal="false"                    timeToIdleSeconds="120"                    timeToLiveSeconds="120"                    overflowToDisk="true"                    maxElementsOnDisk="10000000"                    diskPersistent="false"                    diskExpiryThreadIntervalSeconds="120"                    memoryStoreEvictionPolicy="LRU"                    />        </ehcache>

?

??? ?2.Cache中常用参数的具体意义
??????? (1)name:Cache的唯一标识。
??? ??? (2)maxElementsInMemory:内存中最大缓存对象数。
??? ??? (3)eternal:Element是否永久有效,一旦设置true,timeout将不起作用。
??? ??? (4)timeToIdleSeconds:设置Element在失效前的允许闲置时间。仅当element不是永久有效时使用,可选属性,默认值是0,也就是可闲置时间无穷大。
??? ??? (5)timeToLiveSeconds:设置Element在失效前允许存活时间。最大时间介于创建时间和失效时间之间。仅当element不是永久有效时使用,默认是0.,也就是element存活时间无穷大。
??? ??? (6)overflowToDisk:配置此属性,当内存中Element数量达到maxElementsInMemory时,Ehcache将会Element写到磁盘中。
??? ??? (7)maxElementsOnDisk:磁盘中最大缓存对象数,若是0表示无穷大。
??? ??? (8)?memoryStoreEvictionPolicy:当达到maxElementsInMemory限制时,Ehcache将会根据指定的策略去清理缓存中的内容。默认策略是LRU(最近最少使用),你也可以设置为FIFO(先进先出)或是LFU(较少使用)
???
三.Spring和Ehcache的集成
??? 1.ehcache.xml

<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"            xsi:noNamespaceSchemaLocation="ehcache.xsd">                   <diskStore path="java.io.tmpdir" />                   <defaultCache maxElementsInMemory="10000" eternal="false"                timeToIdleSeconds="600" overflowToDisk="false">            </defaultCache>                   <cache name="levelOneCache" maxElementsInMemory="1000" eternal="false"                timeToIdleSeconds="300" timeToLiveSeconds="1000" overflowToDisk="false" />        </ehcache>

?

??? 2.beans.xml的配置

<bean id="cacheManager"            />            </property>            <property name="cacheName">                <value>configCache</value>            </property>        </bean>

?

??? 3.测试类

       package org.mango.cache.ehcache;               import net.sf.ehcache.Cache;        import net.sf.ehcache.CacheManager;        import net.sf.ehcache.Element;               import org.springframework.beans.factory.BeanFactory;        import org.springframework.beans.factory.xml.XmlBeanFactory;        import org.springframework.core.io.ClassPathResource;        import org.springframework.core.io.Resource;               public class EhcacheTest {                   public static void main(String[] args) {                Resource res = new ClassPathResource("beans.xml");                BeanFactory factory = new XmlBeanFactory(res);                       CacheManager cacheManager = (CacheManager) factory.getBean("cacheManager");                Cache levelOneCache = cacheManager.getCache("levelOneCache");                CacheObject cacheObject = null;                for (int i = 0; i < 10; i++) {                    Element element = levelOneCache.get("key");                           if (element == null) {                        cacheObject = new CacheObject("test");                        element = new Element("key", cacheObject);                        levelOneCache.put(element);                        System.out.println("cacheObject[" + cacheObject + "]" + ",无法从缓存中取到");                    } else {                        cacheObject = (CacheObject) element.getValue();                        System.out.println("cacheObject[" + cacheObject + "]" + ",从缓存中取到");                    }                }            }        }
?

??? ??? 输出如下:

        cacheObject[name:test],无法从缓存中取到        cacheObject[name:test],从缓存中取到        cacheObject[name:test],从缓存中取到        cacheObject[name:test],从缓存中取到        cacheObject[name:test],从缓存中取到   

??? ???
四.利用Spring AOP和Ehcache实现线程级方法缓存
??? 在复杂的业务逻辑或在一次计算中需多次调用同一个DAO或远程服务,在这种情况下,均可对计算结果缓存起来,不但可以减少了不必要的调用次数,还同时可以提高系统运算性能。下面以缓存一个service为例说明一下其用法。
???
??? 1.TestService接口

public interface TestService {                   /**             * 根据userId取得用户名。             *             * @param userId             * @return             */            public String getUserName(String userId);        }
?

???
??? 2.TestServiceImpl实现类

public class TestServiceImpl implements TestService {            /*             * @see org.mango.cache.ehcache.TestService#getUserName(java.lang.String)             */            public String getUserName(String userId) {                return userId;            }        }

?

??? 3.拦截器的实现

public class CacheInterceptor implements MethodInterceptor {                   private Cache cache;                   /**             * @see org.aopalliance.intercept.MethodInterceptor#invoke(org.aopalliance.intercept.MethodInvocation)             */            public Object invoke(MethodInvocation invocation) throws Throwable {                Method method = invocation.getMethod();                String methodName = method.getName();                Object[] arguments = invocation.getArguments();                Object result = invocation.proceed();                       String targetName = method.getDeclaringClass().getName();                String key = getCacheKey(targetName, methodName, arguments);                       Element element = cache.get(key);                       if (element == null) {                           result = invocation.proceed();                    System.out.println("第一次调用方法并缓存其值:" + result);                    cache.put(new Element(key, result));                } else {                    result = element.getValue();                    System.out.println("从缓存中取得的值为:" + result);                }                return result;                   }                   /**             * 生成缓存中的KEY值。             */            protected String getCacheKey(String targetName, String methodName, Object[] arguments) {                StringBuffer sb = new StringBuffer();                sb.append(targetName).append(".").append(methodName);                if ((arguments != null) && (arguments.length != 0)) {                    for (int i = 0; i < arguments.length; i++) {                        sb.append(".").append(arguments[i]);                    }                }                return sb.toString();            }                   public void setCache(Cache cache) {                this.cache = cache;            }               }
?

??? 4.Bean的配置

<bean id="testService" />           <bean id="serviceMethodInterceptor" />            </property>        </bean>           <bean id="serviceAutoProxyCreator"            name="code">public class ServiceTest {           public static void main(String[] args) {                ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");                TestService testService = (TestService) context.getBean("testService");                for (int i = 0; i < 5; i++) {                    testService.getUserName("mango");                }            }        }

?

??? ??? 其输出结果如下:

第一次调用方法并缓存其值:mango从缓存中取得的值为:mango从缓存中取得的值为:mango从缓存中取得的值为:mango从缓存中取得的值为:mango
?

?

1 楼 youjianbo_han_87 2012-08-07   写的不错。。。。 2 楼 szsuyuji 2012-08-07   加上注解 会更好 3 楼 xrythy 2012-08-07   好文章,支持 4 楼 smallbee 2012-08-07   LZ有没有用这个在负载均衡环境下试过,我们之前一个项目也是用这个缓存,但是后来系统用了负载均衡,同步就成了问题,其实ehcache对同步不是很好的,单机还是可以推荐 5 楼 jinnianshilongnian 2012-08-07   smallbee 写道LZ有没有用这个在负载均衡环境下试过,我们之前一个项目也是用这个缓存,但是后来系统用了负载均衡,同步就成了问题,其实ehcache对同步不是很好的,单机还是可以推荐
ehcache适合做jvm内缓存(即在一个jvm内缓存),如果需要集群可以直接使用memcache(分布式缓存)。

尽量不要同步缓存,而是使用分布式。 6 楼 smallbee 2012-08-07   jinnianshilongnian 写道smallbee 写道LZ有没有用这个在负载均衡环境下试过,我们之前一个项目也是用这个缓存,但是后来系统用了负载均衡,同步就成了问题,其实ehcache对同步不是很好的,单机还是可以推荐
ehcache适合做jvm内缓存(即在一个jvm内缓存),如果需要集群可以直接使用memcache(分布式缓存)。

尽量不要同步缓存,而是使用分布式。
分布式?什么意思?缓存同步不是分布式系统的缓存解决办法么? 7 楼 jinnianshilongnian 2012-08-07   smallbee 写道jinnianshilongnian 写道smallbee 写道LZ有没有用这个在负载均衡环境下试过,我们之前一个项目也是用这个缓存,但是后来系统用了负载均衡,同步就成了问题,其实ehcache对同步不是很好的,单机还是可以推荐
ehcache适合做jvm内缓存(即在一个jvm内缓存),如果需要集群可以直接使用memcache(分布式缓存)。

尽量不要同步缓存,而是使用分布式。
分布式?什么意思?缓存同步不是分布式系统的缓存解决办法么?

集群 不同于分布式 先搞清楚概念吧 8 楼 topman007 2012-08-20   "4. 缓存数据会在虚拟机重启的过程中写入磁盘 。"

请问在 web中如何检测 虚拟机重启 了?
谢谢!

热点排行