简介
通常在系统开发中,必不可少的要使用到缓存(Cache),如用户信息、字典信息都会使用缓存来提高性能;但是如何使用好缓存是个需要深入研究的话题,缓存方案没有通用性,针对不同的应用层面,缓存的设计通常也是千差万别的!这里只是介绍了一种比较轻量级、无侵入的缓存方案,该方案基于Spring+SpringModules。
目的
- 方法级别的缓存
- 声明式、无侵入
- 不绑定缓存框架
- JDK 1.4/1.5均适用
实现
基于JDK1.4
JDK1.4中可使用方法映射、元数据(commons-attributes)两种方式声明需要缓存的方法。由于元数据需要导入额外的包,今后JDK升级后无法转换为annotation,且commons-attributes包本身有点BUG,故不在本文讨论之中。
定义缓存实现
目前支持的实现有:
-
org.springmodules.cache.provider.jboss.JbossCacheManagerFactoryBean
-
org.springmodules.cache.provider.jcs.JcsManagerFactoryBean
-
org.springmodules.cache.provider.oscache.OsCacheManagerFactoryBean
-
org.springframework.cache.ehcache.EhCacheManagerFactoryBean
<!-- 缓存实现管理器 --> <bean id="cacheManager" class="org.springframework.cache.ehcache.EhCacheManagerFactoryBean"> </bean>
定义缓存接口
<!-- 缓存统一接口 --> <bean id="cacheProviderFacade" class="org.springmodules.cache.provider.ehcache.EhCacheFacade"> <property name="cacheManager" ref="cacheManager" /> </bean>
定义缓存拦截器
<bean id="cachingInterceptor" class="org.springmodules.cache.interceptor.caching.MethodMapCachingInterceptor"> <property name="cacheProviderFacade" ref="cacheProviderFacade" /> <property name="cachingModels"> <props> <prop key="com.cidp.system.service.IDictService.load*">cacheName=dictCache</prop> </props> </property> </bean>
定义刷新拦截器
<bean id="flushingInterceptor" class="org.springmodules.cache.interceptor.flush.MethodMapFlushingInterceptor"> <property name="cacheProviderFacade" ref="cacheProviderFacade" /> <property name="flushingModels"> <props> <prop key="com.cidp.system.service.IDictService.update*">cacheNames=dictCache</prop> </props> </property> </bean>
为业务层添加缓存拦截器
<!-- 注册自动代理创建,为业务Bean添加事务拦截器 --> <bean id="ServiceAutoProxyCreator" class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator"> <property name="proxyTargetClass" value="false"></property> <property name="beanNames"> <list> <value>*Service</value> </list> </property> <property name="interceptorNames"> <list> <value>cachingInterceptor</value> <value>flushingInterceptor</value> </list> </property> </bean>
基于JDK1.5
JDK 1.5中出现了annotation这样的好东西,使得我们可以在方法级别上做更精细的控制,将基于XML配置转入源代码中。
注:缓存实现和缓存接口配置同上,拦截器绑定同上
定义缓存拦截器
<bean id="cachingAttributeSource" class="org.springmodules.cache.annotations.AnnotationCachingAttributeSource"> </bean> <bean id="cachingInterceptor" class="org.springmodules.cache.interceptor.caching.MetadataCachingInterceptor"> <property name="cacheProviderFacade" ref="cacheProviderFacade" /> <property name="cachingAttributeSource" ref="cachingAttributeSource" /> <property name="cachingModels"> <props> <prop key="dictCaching">cacheName=dictCache</prop> </props> </property> </bean>
定义刷新拦截器
<bean id="flushingAttributeSource" class="org.springmodules.cache.annotations.AnnotationFlushingAttributeSource"> </bean> <bean id="flushingInterceptor" class="org.springmodules.cache.interceptor.flush.MetadataCachingInterceptor"> <property name="cacheProviderFacade" ref="cacheProviderFacade" /> <property name="flushingAttributeSource" ref="flushingAttributeSource" /> <property name="flushingModels"> <props> <prop key="dictFlushing">cacheNames=dictCache</prop> </props> </property> </bean>
Java代码
@Cacheable(modelId = "dictCaching")
public Dict load(Long id);
@CacheFlush(modelId = "dictFlushing")
public void update(Dict dict);
ehcache
springmodules支持<ehcache>标签形式的配置方式,可简化以上用<bean>定义拦截器的方式。如果项目中使用的缓存框架为ehcache,可通过该标签简化配置