Java开发框架spring实现自定义缓存标签

 更新时间:2015年12月14日 16:35:35   作者:txxs  
这篇文章主要介绍了Java开发框架spring实现自定义缓存标签的详细代码,感兴趣的小伙伴们可以参考一下

自从spring3.1之后,spring引入了抽象缓存,可以通过在方法上添加@Cacheable等标签对方法返回的数据进行缓存。但是它到底是怎么实现的呢,我们通过一个例子来看一下。首先我们定义一个@MyCacheable

package caching.springaop; 
 
import java.lang.annotation.Retention; 
import java.lang.annotation.RetentionPolicy; 
import java.lang.annotation.Target; 
import java.lang.annotation.ElementType; 
 
/** 
 * 使用@MyCacheable注解方法 
 */ 
@Retention(RetentionPolicy.RUNTIME) 
@Target(ElementType.METHOD) 
public @interface MyCacheable{ 
 
} 

然后定义处理MyCacheable的切面

package caching.springaop; 
 
import java.util.HashMap; 
import java.util.Map; 
 
import org.apache.log4j.Logger; 
import org.aspectj.lang.ProceedingJoinPoint; 
import org.aspectj.lang.annotation.Around; 
import org.aspectj.lang.annotation.Aspect; 
import org.aspectj.lang.annotation.Pointcut; 
 
/** 
 * 处理MyCacheable方法的切面 
 */ 
@Aspect 
public class CacheAspect { 
 
  private Logger logger = Logger.getLogger(CacheAspect.class); 
  private Map<String, Object> cache; 
 
  public CacheAspect() { 
    cache = new HashMap<String, Object>(); 
  } 
 
  /** 
   * 所有标注了@Cacheable标签的方法切入点 
   */ 
  @Pointcut("execution(@MyCacheable * *.*(..))") 
  @SuppressWarnings("unused") 
  private void cache() { 
  } 
 
  @Around("cache()") 
  public Object aroundCachedMethods(ProceedingJoinPoint thisJoinPoint) 
      throws Throwable { 
    logger.debug("Execution of Cacheable method catched"); 
    //产生缓存数据的key值,像是这个样子caching.aspectj.Calculator.sum(Integer=1;Integer=2;) 
    StringBuilder keyBuff = new StringBuilder(); 
    //增加类的名字 
    keyBuff.append(thisJoinPoint.getTarget().getClass().getName()); 
    //加上方法的名字 
    keyBuff.append(".").append(thisJoinPoint.getSignature().getName()); 
    keyBuff.append("("); 
    //循环出cacheable方法的参数 
    for (final Object arg : thisJoinPoint.getArgs()) { 
      //增加参数的类型和值 
      keyBuff.append(arg.getClass().getSimpleName() + "=" + arg + ";"); 
    } 
    keyBuff.append(")"); 
    String key = keyBuff.toString(); 
    logger.debug("Key = " + key); 
    Object result = cache.get(key); 
    if (result == null) { 
      logger.debug("Result not yet cached. Must be calculated..."); 
      result = thisJoinPoint.proceed(); 
      logger.info("Storing calculated value '" + result + "' to cache"); 
      cache.put(key, result); 
    } else { 
      logger.debug("Result '" + result + "' was found in cache"); 
     
    return result; 
  } 
 
} 

上述代码展示了如何处理MyCacheable自定义的标签,以及默认情况下产生key值的规则。最后生成的key值大概是这个样子:caching.aspectj.Calculator.sum(Integer=1;Integer=2;)
下边这段代码在方法上添加了MyCacheable标签

package caching.springaop; 
 
import org.apache.log4j.Logger; 
public class Calculator { 
  private Logger logger = Logger.getLogger(Calculator.class); 
  @MyCacheable 
  public int sum(int a, int b) { 
    logger.info("Calculating " + a + " + " + b); 
    try { 
      //假设这是代价非常高的计算 
      Thread.sleep(3000); 
    } catch (InterruptedException e) { 
      logger.error("Something went wrong...", e); 
    } 
    return a + b; 
  } 
} 

在方法上加了MyCacheable标签,当key值相同的情况下会直接在缓存中获取数据,如果没有相同的key值,则会重新计算,因为这里只是一个加和操作,耗时非常的短暂。我们在这里让其睡眠3秒钟。
我们在spring-config.xml配置如下:

<?xml version="1.0" encoding="UTF-8"?> 
<beans xmlns="http://www.springframework.org/schema/beans" 
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
  xmlns:aop="http://www.springframework.org/schema/aop" 
  xsi:schemaLocation="http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd 
    http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd"> 
  <aop:aspectj-autoproxy /> 
  <bean class="caching.springaop.CacheAspect" /> 
  <bean id="calc" class="caching.springaop.Calculator" /> 
</beans> 

测试类:

package caching.springaop; 
 
import org.apache.log4j.Logger; 
import org.springframework.context.ApplicationContext; 
import org.springframework.context.support.ClassPathXmlApplicationContext; 
 
/** 
 * 使用SpringAOP缓存的简单例子 
 * @author txxs 
 */ 
public class App { 
 
  private static Logger logger = Logger.getLogger(App.class); 
 
  public static void main(String[] args) { 
    logger.debug("Starting..."); 
    ApplicationContext ctx = new ClassPathXmlApplicationContext("spring-config.xml"); 
    Calculator calc = (Calculator) ctx.getBean("calc"); 
    //计算出来的结果将会被存储在cache 
    logger.info("1 + 2 = " + calc.sum(1, 2)); 
    //从缓存中获取结果 
    logger.info("1 + 2 = " + calc.sum(1, 2)); 
    logger.debug("Finished!"); 
  } 
 
} 

我们看一下运行的结果:

从结果来看第一次直接计算结果,第二次从缓存中获取。

以上就是spring实现自定义缓存标签的全部内容,希望对大家的学习有所帮助

相关文章

  • Java注册邮箱激活验证实现代码

    Java注册邮箱激活验证实现代码

    这篇文章主要介绍了Java注册邮箱激活验证实现代码,有需要的朋友可以参考一下
    2013-12-12
  • 解决Mybatis mappe同时传递 List 和其他参数报错的问题

    解决Mybatis mappe同时传递 List 和其他参数报错的问题

    在使用MyBatis时,如果需要传递多个参数到SQL中,可以遇到参数绑定问题,解决方法包括使用@Param注解和修改mapper.xml配置,感兴趣的朋友跟随小编一起看看吧
    2024-09-09
  • Java中Map与对象之间互相转换的几种常用方式

    Java中Map与对象之间互相转换的几种常用方式

    在Java中将对象和Map相互转换是常见的操作,可以通过不同的方式实现这种转换,下面这篇文章主要给大家介绍了关于Java中Map与对象之间互相转换的几种常用方式,需要的朋友可以参考下
    2024-01-01
  • Spring Bean创建流程分析讲解

    Spring Bean创建流程分析讲解

    很多时候我们需要根据不同的条件在容器中加载不同的Bean,或者根据不同的条件来选择是否在容器中加载某个Bean,这就是Bean的加载控制,一般我们可以通过编程式或注解式两种不同的方式来完成Bean的管理
    2023-01-01
  • Spring Data Jpa 自动生成表结构的方法示例

    Spring Data Jpa 自动生成表结构的方法示例

    这篇文章主要介绍了Spring Data Jpa 自动生成表结构的方法示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-04-04
  • Java hashCode() 方法详细解读

    Java hashCode() 方法详细解读

    Java.lang.Object 有一个hashCode()和一个equals()方法,这两个方法在软件设计中扮演着举足轻重的角色,本文对hashCode()方法深入理解,希望能帮助大家
    2016-07-07
  • java实现多线程卖票功能

    java实现多线程卖票功能

    这篇文章主要为大家详细介绍了java实现多线程卖票功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-07-07
  • Java如何接收XML格式参数并转换为JSON

    Java如何接收XML格式参数并转换为JSON

    在 Java 应用程序中,处理 XML 数据并将其转换为 JSON 格式是很常见的任务,这篇文章为大家整理了一下具体的实现方法,希望对大家有所帮助
    2025-03-03
  • 总结java多线程之互斥与同步解决方案

    总结java多线程之互斥与同步解决方案

    文中总结了线程互斥与同步,synchronized使用细节及原理,Reentrylock使用细节等知识,对解决Java多线程互斥与同步等问题很有效,,需要的朋友可以参考下
    2021-05-05
  • springBoot @Scheduled实现多个任务同时开始执行

    springBoot @Scheduled实现多个任务同时开始执行

    这篇文章主要介绍了springBoot @Scheduled实现多个任务同时开始执行,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-12-12

最新评论