您的位置:首页 > 编程语言 > Java开发

Java缓存框架使用EhCache结合Spring AOP

2017-07-17 16:59 417 查看
一.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片段:

Java代码

<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"
class="org.springframework.cache.ehcache.EhCacheManagerFactoryBean">
<property name="configLocation">
<value>classpath:ehcache.xml</value>
</property>
</bean>

<bean id="levelOneCache" class="org.springframework.cache.ehcache.EhCacheFactoryBean">
<property name="cacheManager">
<ref local="cacheManager" />
</property>
<property name="cacheName">
<value>configCache</value>
</property>
</bean>


3.测试类

Java
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" class="org.mango.cache.ehcache.TestServiceImpl" />

<bean id="serviceMethodInterceptor" class="org.mango.cache.ehcache.CacheInterceptor">
<property name="cache">
<ref local="levelOneCache" />
</property>
</bean>

<bean id="serviceAutoProxyCreator"
class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
<property name="interceptorNames">
<list>
<value>serviceMethodInterceptor</value>
</list>
</property>
<property name="beanNames">
<value>*Service</value>
</property>
</bean>


5.测试方法

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");
}
}
}
其输出结果如下:[/code]

第一次调用方法并缓存其值:mango
从缓存中取得的值为:mango
从缓存中取得的值为:mango
从缓存中取得的值为:mango
从缓存中取得的值为:mango
来源:https://my.oschina.net/iblike/blog/71260
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: