一般缓存与数据库的配合使用是这样的。

1.查询缓存中是否有数据。

2.缓存中无数据,查询数据库。

3.把数据库数据插入到缓存中。

  其实我们发现 1,3 都是固定的套路,只有2 是真正的业务代码。我们可以把1,3 抽取出来,封装到一个自定义注解@myCache 上,通过给2方法加一个注解,实现代码的解耦。

package com.itbac.common.cache;

import org.springframework.stereotype.Service;

@Service
public class SkuQueryService {
   //注解的使用
@myCache(key = "'SkuQueryService_findById' + #id")
public Object findById(String id){ System.out.println("findById方法查询数据库"); return "数据库返回值:"+id;
}
}

自定义注解

package com.itbac.common.cache;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; // 注解的生命周期:运行时
@Retention(RetentionPolicy.RUNTIME)
// 注解的应用范围:修饰方法
@Target(ElementType.METHOD)
public @interface myCache {
/**
* key 的生成规则,通过springEL表达式
* @return
*/
String key();
}

AOP切面类,切面编程,动态解析注解。

package com.itbac.common.cache;

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.DefaultParameterNameDiscoverer;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.expression.Expression;
import org.springframework.expression.spel.standard.SpelExpressionParser;
import org.springframework.expression.spel.support.StandardEvaluationContext;
import org.springframework.stereotype.Component; import java.lang.reflect.Method;
import java.util.concurrent.TimeUnit; /**
* Aop切面编程
*/
@Component
@Aspect
public class cacheAOP { @Autowired
private RedisTemplate redisTemplate;
  
  
   // 环绕通知 : 监控自定义注解 。 
@Around("@annotation(com.itbac.common.cache.myCache)")
public Object doAnyThing(ProceedingJoinPoint joinPoint) throws Throwable {
String key =null;
//1.反射技术:从注解里里面,读取key的生成规则。
//1.1 从切入点,获取方法签名
MethodSignature signature = (MethodSignature) joinPoint.getSignature();
//1.2 从切点,获取目标对象的字节码,的方法。参数:(方法名,方法的所有参数)。
Method method = joinPoint.getTarget().getClass().getMethod(signature.getName(), signature.getMethod().getParameterTypes());
//1.3 从方法获取注解。
myCache annotation = method.getAnnotation(myCache.class);
//1.4 从注解,获取注解信息。'SkuQueryService_findById' + #id
String keyEL = annotation.key(); //2. 创建 springEL表达式 解析器
SpelExpressionParser parser = new SpelExpressionParser();
// 解析器 获取指定表达式 'SkuQueryService_findById' + #id 的表达式对象
Expression expression = parser.parseExpression(keyEL);
// 设置解析上下文
StandardEvaluationContext context = new StandardEvaluationContext();
//2.1 创建默认参数名 发现者
DefaultParameterNameDiscoverer discoverer = new DefaultParameterNameDiscoverer();
//2.2 获取方法中的所有参数名。
String[] parameterNames = discoverer.getParameterNames(method);
//2.3 获取切点方法中的所有参数值。
Object[] args = joinPoint.getArgs();
for (int i = 0; i < parameterNames.length; i++) {
// 把参数名,参数值,设置到解析器上下文
context.setVariable(parameterNames[i],args[i].toString());
}
//表达式 匹配 解析上下文 中的内容 ,拿到key
key = expression.getValue(context).toString(); Object o = redisTemplate.opsForValue().get(key);
if (o !=null){
// 缓存中有数据,直接返回。
//查询缓存
System.out.println("查询缓存返回。");
//延迟缓存失效时间,1天。
redisTemplate.expire(key,1,TimeUnit.DAYS);
return o;
}
// 缓存穿透标记
Object penetrateFlag = redisTemplate.opsForValue().get(key + "penetrateFlag");
if (null == penetrateFlag){
// 没有防止 缓存穿透标记,查数据库。 // 执行切点 中的代码,查询数据库。
Object proceed = joinPoint.proceed(); if (null == proceed){
// 数据库数据为空,设置缓存穿透标记。15分钟
redisTemplate.opsForValue().set(key + "penetrateFlag",true,15,TimeUnit.MINUTES);
}else {
// 数据库数据不为空,把数据存到缓存中。缓存1天。
redisTemplate.opsForValue().set(key,proceed,1, TimeUnit.DAYS);
}
return proceed ;
}
// 延迟缓存失效时间 15分钟 缓存穿透标记
redisTemplate.expire(key+"penetrateFlag",15,TimeUnit.MINUTES);
// 返回
return null;
}
}

这样就可以通过一个自定义注解@myCache ,实现了缓存与业务代码的解耦。其中还包含了防止缓存穿透的使用技巧。不要告诉别人哦。

注解与AOP切面编程实现redis缓存与数据库查询的解耦的更多相关文章

  1. SpringBoot 通过自定义注解实现AOP切面编程实例

    一直心心念的想写一篇关于AOP切面实例的博文,拖更了许久之后,今天终于着手下笔将其完成. 基础概念 1.切面(Aspect) 首先要理解‘切’字,需要把对象想象成一个立方体,传统的面向对象变成思维,类 ...

  2. 注解配置AOP切面编程

    1.导入先关jar包 2.编写applicationContext.xml,配置开启注解扫描和切面注解扫描 <?xml version="1.0" encoding=&quo ...

  3. 基于SpringBoot AOP面向切面编程实现Redis分布式锁

    基于SpringBoot AOP面向切面编程实现Redis分布式锁 基于SpringBoot AOP面向切面编程实现Redis分布式锁 基于SpringBoot AOP面向切面编程实现Redis分布式 ...

  4. AOP切面编程在android上的应用

    代码地址如下:http://www.demodashi.com/demo/12563.html 前言 切面编程一直是一个热点的话题,这篇文章讲讲一个第三方aop库在android上的应用.第三方AOP ...

  5. SpringBoot2.0 基础案例(11):配置AOP切面编程,解决日志记录业务

    本文源码 GitHub地址:知了一笑 https://github.com/cicadasmile/spring-boot-base 一.AOP切面编程 1.什么是AOP编程 在软件业,AOP为Asp ...

  6. 十:SpringBoot-配置AOP切面编程,解决日志记录业务

    SpringBoot-配置AOP切面编程,解决日志记录业务 1.AOP切面编程 1.1 AOP编程特点 1.2 AOP中术语和图解 2.SpringBoot整合AOP 2.1 核心依赖 2.2 编写日 ...

  7. Spring MVC通过AOP切面编程 来拦截controller 实现日志的写入

    首选需要参考的是:[参考]http://www.cnblogs.com/guokai870510826/p/5977948.html    http://www.cnblogs.com/guokai8 ...

  8. Spring AOP 切面编程记录日志和接口执行时间

    最近客户现在提出系统访问非常慢,需要优化提升访问速度,在排查了nginx.tomcat内存和服务器负载之后,判断是数据库查询速度慢,进一步排查发现是因为部分视图和表查询特别慢导致了整个系统的响应时间特 ...

  9. 本地缓存,Redis缓存,数据库DB查询(结合代码分析)

    问题背景 为什么要使用缓存?本地缓存/Redis缓存/数据库查询优先级? 一.为什么要使用缓存 原因:CPU的速度远远高于磁盘IO的速度问题:很多信息存在数据库当中的,每次查询数据库就是一次IO操作所 ...

随机推荐

  1. esxi开启SSH

  2. Effective Java - 静态方法与构造器

    目录 用静态工厂方法替代构造器? 静态工厂有名称 静态工厂不必重新创建一个对象 静态工厂可以返回任何子类型对象 静态工厂返回的类可以动态变化 静态工厂返回的类可以不存在 静态工厂方法的缺点 静态工厂方 ...

  3. 聚类时的轮廓系数评价和inertia_

    在进行聚类分析时,机器学习库中提供了kmeans++算法帮助训练,然而,根据不同的问题,需要寻找不同的超参数,即寻找最佳的K值 最近使用机器学习包里两个内部评价聚类效果的方法:clf=KMeans(n ...

  4. 关于网页授权access_token和普通access_token的区别

    关于网页授权access_token和普通access_token的区别 1.微信网页授权是通过OAuth2.0机制实现的,在用户授权给公众号后,公众号可以获取到一个网页授权特有的接口调用凭证(网页授 ...

  5. PCB 板边倒圆角的实现方法(基本算法一)

    PCB外形是直角时外形时,通常工程制作时,外是直角或尖角的地方倒圆角,主要是为了防止板边容易划伤板且容易扎伤人 所以当客户没有特殊要求时,PCB外形是直角时一般会默认倒角0.5mm圆角(如下图所示) ...

  6. 宏旺半导体浅谈存储芯片LPDDR4X与UFS2.1的差别

    现在市面上手机参数动不动就是8GB+128GB,手机的这些参数是越大越好吗?这些数字代表什么?宏旺半导体ICMAX给大家科普下. 手机的运行内存RAM——LPDDR4X LPDDR4X为RAM(运存) ...

  7. springcloud高可用方案

    1. 场景描述 公司在规划后续可能会做中台服务,考虑用微服务的方案,让用springcloud部署个简单的高可用Demo. 2. 解决方案 2.1 方案说明 demo用了5台虚拟机: (1)1台gat ...

  8. 跟着大彬读源码 - Redis 4 - 服务器的事件驱动有什么含义?(上)

    众所周知,Redis 服务器是一个事件驱动程序.那么事件驱动对于 Redis 而言有什么含义?源码中又是如何实现事件驱动的呢?今天,我们一起来认识下 Redis 服务器的事件驱动. 对于 Redis ...

  9. 【DFS的分支限界】(例题-算式等式)

    不知道DFS的请滚去 这里瞅一眼再说. -分支限界- 基本概念: 类似于回溯法,也是一种在问题的解空间树T上搜索问题解的算法.但在一般情况下,分支限界法与回溯法的求解目标不同.回溯法的求解目标是找出T ...

  10. 题解 P5016 【龙虎斗】

    首先祝各位大佬noip有个好成绩吧 当时比赛有个大数据,蒟蒻我暴力居然过了,好激动 这题一定要注意开long long (那个大数据就是我开long long才过的) 还有刚开始应设置答案为m(见解析 ...