当前位置: 首页 > 编程笔记 >

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

颜霖
2023-03-14
本文向大家介绍Java开发框架spring实现自定义缓存标签,包括了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实现自定义缓存标签的全部内容,希望对大家的学习有所帮助

 类似资料:
  • 我对spring boot中的缓存技术很陌生,我在我的项目中使用了缓存。现在我有一个新的需求,需要设置缓存TTL,并且TTL必须在密钥基础上扩展。 E、 g.如果我将缓存过期时间设置为5分钟,则在缓存过期之前,必须检查传入密钥是否存在于缓存中。如果传入密钥存在于缓存中,则TTL必须延长5分钟,否则应清除或收回缓存,并创建新的缓存。 我希望我清楚我的要求,请提供我的解决方案与完美的例子,适合我的要求

  • 在Spring有什么方法可以达到同样的效果。 我确实在这里查看了@cacheable http://www.baeldung.com/spring-cache-tutorial 但是我无法理解我是如何使spring使用我的“memcached”类的,更具体地说,我在哪里连接我的“memcached”类,以便它可用于上面示例中的类“abc

  • 我使用的是版本。 我已经使用service Builder为portlet创建了自定义实体。我想缓存自定义实体。 我在我的portal-ext.properties中设置了以下属性以启用缓存。 我创建了ehcache.xml文件来覆盖ehcache-failsafe.xml来配置我的自定义实体,以便它能够启用缓存。 我的ehcache.xml文件在我的类路径[classpath:liferay-p

  • 我们正在使用firebase实时数据库,我正在考虑在本地实现一个缓存来减少重复调用。

  • 本文向大家介绍Android自定义TitleView标题开发实例,包括了Android自定义TitleView标题开发实例的使用技巧和注意事项,需要的朋友参考一下 Android开发过程中,经常遇到一个项目需要重复的定义相同样式的标题栏,Android相继推出了actionBar, toolBar, 相信有用到的朋友也会遇到一些不如意的时候,比如标题栏居中时,需要自定义xml文件给toolBar等

  • 自定义state缓存 TIP starter 内置了2种缓存实现 一种是上一节演示的默认实现 另一种是用户自定义的扩展实现。 本节将会使用JustAuth-demo (opens new window)进行演示扩展Redis缓存的方式,当然了,你也可以自定义实现你自己的缓存。 添加 Redis 依赖 <dependency> <groupId>org.springframework.boo