通过AOP方式实现Service计算结果的缓存

2024-05-02 02:08

本文主要是介绍通过AOP方式实现Service计算结果的缓存,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

AOP为Aspect Oriented Programming, 面向切面的编程。意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。

Spring框架的AOP基础知识详见:

Spring实现AOP的4种方式

Spring AOP 详解

AOP的应用场景很多。可以作为日志系统的记录拦截,也可以用于缓存的拦截。

以下用我自己的框架项目 springMVC 中实现的Ehcache对Service层计算结果缓存的实现为例子。介绍一下。

Service缓存实现

主要实现思路通过AOP拦截的方式插入缓存层的操作。

参考:Spring AOP+ehCache简单缓存系统解决方案

简单的配置以及实现效果

Ehcache配置

ehcache的配置文件默认在 classpath:ehcache.xml MAVEN工程对应 src/main/resources 。定义了缓存的配置信息,* 必须有一个 默认缓存defaultCache *。在管理器找不到缓存的时候,就会用默认缓存。


<?xml version="1.0" encoding="UTF-8"?>
<!-- Ehcache2.x的变化(取自https://github.com/springside/springside4/wiki/Ehcache) -->
<!-- 1)最好在ehcache.xml中声明不进行updateCheck -->
<!-- 2)为了配合BigMemory和Size Limit,原来的属性最好改名 -->
<!-- maxElementsInMemory->maxEntriesLocalHeap -->
<!-- maxElementsOnDisk->maxEntriesLocalDisk -->
<ehcache mlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd"><diskStore path="java.io.tmpdir" /><defaultCache maxElementsInMemory="1000" eternal="false"timeToIdleSeconds="120" timeToLiveSeconds="120" overflowToDisk="false" /><cache name="myCache" maxElementsOnDisk="20000"maxElementsInMemory="2000" eternal="true" overflowToDisk="true"diskPersistent="true" />
</ehcache>
<!-- <diskStore>==========当内存缓存中对象数量超过maxElementsInMemory时,将缓存对象写到磁盘缓存中(需对象实现序列化接口) <diskStore path="">==用来配置磁盘缓存使用的物理路径,Ehcache磁盘缓存使用的文件后缀名是*.data和*.index name=================缓存名称,cache的唯一标识(ehcache会把这个cache放到HashMap里) maxElementsOnDisk====磁盘缓存中最多可以存放的元素数量,0表示无穷大 maxElementsInMemory==内存缓存中最多可以存放的元素数量,若放入Cache中的元素超过这个数值,则有以下两种情况 1)若overflowToDisk=true,则会将Cache中多出的元素放入磁盘文件中 2)若overflowToDisk=false,则根据memoryStoreEvictionPolicy策略替换Cache中原有的元素 eternal==============缓存中对象是否永久有效,即是否永驻内存,true时将忽略timeToIdleSeconds和timeToLiveSeconds timeToIdleSeconds====缓存数据在失效前的允许闲置时间(单位:秒),仅当eternal=false时使用,默认值是0表示可闲置时间无穷大,此为可选属性 即访问这个cache中元素的最大间隔时间,若超过这个时间没有访问此Cache中的某个元素,那么此元素将被从Cache中清除 timeToLiveSeconds====缓存数据在失效前的允许存活时间(单位:秒),仅当eternal=false时使用,默认值是0表示可存活时间无穷大 即Cache中的某元素从创建到清楚的生存时间,也就是说从创建开始计时,当超过这个时间时,此元素将从Cache中清除 overflowToDisk=======内存不足时,是否启用磁盘缓存(即内存中对象数量达到maxElementsInMemory时,Ehcache会将对象写到磁盘中) 会根据标签中path值查找对应的属性值,写入磁盘的文件会放在path文件夹下,文件的名称是cache的名称,后缀名是data diskPersistent=======是否持久化磁盘缓存,当这个属性的值为true时,系统在初始化时会在磁盘中查找文件名为cache名称,后缀名为index的文件 这个文件中存放了已经持久化在磁盘中的cache的index,找到后会把cache加载到内存 要想把cache真正持久化到磁盘,写程序时注意执行net.sf.ehcache.Cache.put(Element element)后要调用flush()方法 diskExpiryThreadIntervalSeconds==磁盘缓存的清理线程运行间隔,默认是120秒 diskSpoolBufferSizeMB============设置DiskStore(磁盘缓存)的缓存区大小,默认是30MB memoryStoreEvictionPolicy========内存存储与释放策略,即达到maxElementsInMemory限制时,Ehcache会根据指定策略清理内存 共有三种策略,分别为LRU(最近最少使用)、LFU(最常用的)、FIFO(先进先出) -->

AOP拦截

拦截器编写
  • 取缓存 MethodCacheInterceptor

package org.company.core.common.interceptor;import java.io.Serializable;
import java.lang.reflect.Method;
import java.util.List;import net.sf.ehcache.Cache;
import net.sf.ehcache.Element;import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.util.Assert;public class MethodCacheInterceptor implements MethodInterceptor,InitializingBean {private Cache cache;/*** 设置缓存名*/public void setCache(Cache cache) {this.cache = cache;}/*** 检查是否提供必要参数。*/public void afterPropertiesSet() throws Exception {Assert.notNull(cache,"A cache is required. Use setCache(Cache) to provide one.");}/*** 主方法 如果某方法可被缓存就缓存其结果 方法结果必须是可序列化的(serializable)*/public Object invoke(MethodInvocation invocation) throws Throwable {
//      String methodName = invocation.getMethod().getName();
//      if(methodName.contains("find")){
//          return get(invocation);
//      } else {
//          return remove(invocation);
//      }String targetName = invocation.getThis().getClass().getName();String methodName = invocation.getMethod().getName();Object[] arguments = invocation.getArguments();Object result;String cacheKey = getCacheKey(targetName, methodName, arguments);System.out.println("getting from cache...");Element element = cache.get(cacheKey);System.out.println("cache Key:" + cacheKey);if (element == null) {System.out.println("miss");result = invocation.proceed();element = new Element(cacheKey, (Serializable) result);cache.put(element);return element.getValue();} else {System.out.println("hit");return element.getValue();}}//  }/*** creates cache key: targetName.methodName.argument0.argument1...*/private 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();}
}
  • 删除缓存

package org.company.core.common.interceptor;import java.lang.reflect.Method;
import java.util.List;import net.sf.ehcache.Cache;import org.springframework.aop.AfterReturningAdvice;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.util.Assert;public class MethodCacheAfterAdvice implements AfterReturningAdvice, InitializingBean    
{    private Cache cache;    public void setCache(Cache cache) {    this.cache = cache;    }    public MethodCacheAfterAdvice() {    super();    }    @Transactionalpublic void afterReturning(Object arg0, Method arg1, Object[] arg2, Object arg3) throws Throwable {    String className = arg3.getClass().getName();    List list = cache.getKeys();    System.out.println("invoking...");System.out.println("removing cache key:"+className+".*");for(int i = 0;i<list.size();i++){    String cacheKey = String.valueOf(list.get(i));    if(cacheKey.startsWith(className)){    System.out.println("remving key:"+cacheKey);cache.remove(cacheKey);    }    }    }    public void afterPropertiesSet() throws Exception {    Assert.notNull(cache, "Need a cache. Please use setCache(Cache) create it.");    }      
}   
配置文件

配置详见 applicationContext-ehcache.xml

注册拦截器类

声明了拦截器的BEAN

<!-- find/create cache拦截器 -->  <bean id="methodCacheInterceptor" class="org.company.frame.interceptor.MethodCacheInterceptor"><property name="cache"><ref local="demoCache" /></property></bean><!-- flush cache拦截器 -->  <bean id="methodCacheAfterAdvice" class="org.company.frame.interceptor.MethodCacheAfterAdvice">  <property name="cache">  <ref local="demoCache" />  </property>  </bean>  
切入点声明

将拦截器BEAN与方法匹配。

<!-- find/get cache拦截器方法匹配 --><bean id="methodCachePointCut"class="org.springframework.aop.support.RegexpMethodPointcutAdvisor"><property name="advice"><ref local="methodCacheInterceptor" /></property><property name="patterns"><list><value>org.company.core.*.service.*find.*</value></list></property></bean><!-- flush 拦截器方法匹配  --><bean id="methodCachePointCutAdvice" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor">  <property name="advice">  <ref local="methodCacheAfterAdvice"/>  </property>  <property name="patterns">  <list>  <value>org.company.core.*.service.*create.*</value>  <value>org.company.core.*.service.*update.*</value>  <value>org.company.core.*.service.*delete.*</value>  </list>  </property>  </bean>
AOP的最终声明

绑定服务BEAN与拦截器切入点。

<!-- AOP 的最终配置 --><!-- BEAN与find/get 拦截器关联 --><bean id="myService" class="org.springframework.aop.framework.ProxyFactoryBean"><property name="target"><ref local="testTableServiceBean" /></property><property name="interceptorNames"><list><value>methodCachePointCut</value><value>methodCachePointCutAdvice</value></list></property></bean> <!-- 拦截BEAN --><bean id="testTableServiceBean" class="org.company.core.moduel.service.TestTableServiceImpl"></bean>

缓存的最终实现效果如图:

这里写图片描述

这篇关于通过AOP方式实现Service计算结果的缓存的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/953178

相关文章

Vue项目的甘特图组件之dhtmlx-gantt使用教程和实现效果展示(推荐)

《Vue项目的甘特图组件之dhtmlx-gantt使用教程和实现效果展示(推荐)》文章介绍了如何使用dhtmlx-gantt组件来实现公司的甘特图需求,并提供了一个简单的Vue组件示例,文章还分享了一... 目录一、首先 npm 安装插件二、创建一个vue组件三、业务页面内 引用自定义组件:四、dhtmlx

linux打包解压命令方式

《linux打包解压命令方式》文章介绍了Linux系统中常用的打包和解压命令,包括tar和zip,使用tar命令可以创建和解压tar格式的归档文件,使用zip命令可以创建和解压zip格式的压缩文件,每... 目录Lijavascriptnux 打包和解压命令打包命令解压命令总结linux 打包和解压命令打

Vue ElementUI中Upload组件批量上传的实现代码

《VueElementUI中Upload组件批量上传的实现代码》ElementUI中Upload组件批量上传通过获取upload组件的DOM、文件、上传地址和数据,封装uploadFiles方法,使... ElementUI中Upload组件如何批量上传首先就是upload组件 <el-upl

Docker部署Jenkins持续集成(CI)工具的实现

《Docker部署Jenkins持续集成(CI)工具的实现》Jenkins是一个流行的开源自动化工具,广泛应用于持续集成(CI)和持续交付(CD)的环境中,本文介绍了使用Docker部署Jenkins... 目录前言一、准备工作二、设置变量和目录结构三、配置 docker 权限和网络四、启动 Jenkins

Python3脚本实现Excel与TXT的智能转换

《Python3脚本实现Excel与TXT的智能转换》在数据处理的日常工作中,我们经常需要将Excel中的结构化数据转换为其他格式,本文将使用Python3实现Excel与TXT的智能转换,需要的可以... 目录场景应用:为什么需要这种转换技术解析:代码实现详解核心代码展示改进点说明实战演练:从Excel到

Python中常用的四种取整方式分享

《Python中常用的四种取整方式分享》在数据处理和数值计算中,取整操作是非常常见的需求,Python提供了多种取整方式,本文为大家整理了四种常用的方法,希望对大家有所帮助... 目录引言向零取整(Truncate)向下取整(Floor)向上取整(Ceil)四舍五入(Round)四种取整方式的对比综合示例应

如何使用CSS3实现波浪式图片墙

《如何使用CSS3实现波浪式图片墙》:本文主要介绍了如何使用CSS3的transform属性和动画技巧实现波浪式图片墙,通过设置图片的垂直偏移量,并使用动画使其周期性地改变位置,可以创建出动态且具有波浪效果的图片墙,同时,还强调了响应式设计的重要性,以确保图片墙在不同设备上都能良好显示,详细内容请阅读本文,希望能对你有所帮助...

C# string转unicode字符的实现

《C#string转unicode字符的实现》本文主要介绍了C#string转unicode字符的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随... 目录1. 获取字符串中每个字符的 Unicode 值示例代码:输出:2. 将 Unicode 值格式化

python安装whl包并解决依赖关系的实现

《python安装whl包并解决依赖关系的实现》本文主要介绍了python安装whl包并解决依赖关系的实现,文中通过图文示例介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录一、什么是whl文件?二、我们为什么需要使用whl文件来安装python库?三、我们应该去哪儿下

Python脚本实现图片文件批量命名

《Python脚本实现图片文件批量命名》这篇文章主要为大家详细介绍了一个用python第三方库pillow写的批量处理图片命名的脚本,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录前言源码批量处理图片尺寸脚本源码GUI界面源码打包成.exe可执行文件前言本文介绍一个用python第三方库pi