简介
通常在系统开发中,必不可少的要使用到缓存(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,可通过该标签简化配置