模拟springcache 注解实现 缓存操作

模拟springcache 注解实现 缓存操作

背景就是 每次都需要查询判断 不为空 返回 缓存里面数据,其实有点重复代码

提供两种方式
  1. 直接使用springcache

  2. 自定义注解 + aop 实现

    2.1 定义两个注解

    2.2 实现切面 获取注解里面的值进行对应的逻辑判断

    Retention(RetentionPolicy.RUNTIME)
    @Target({ElementType.METHOD})
    @Documented
    public @interface CacheKey {
    
        /**
         * 缓存键名称注解,默认值为类名称
         *
         * @return
         */
        String key() default "cacheKey:";
    
        /**
         * 缓存过期时间,默认3分钟
         *
         * @return
         */
        int expireTime() default 3*60;
    
        /**
         * 缓存过期时间单位,默认为分钟
         *
         * @return
         */
        TimeUnit timeUnit() default TimeUnit.SECONDS;
    
        /**
         * 缓存事件 默认查询
         * @return
         */
        CacheEvent cacheEvent() default CacheEvent.QUERY;
    
    }
    
    /**
     * 参数注解,用于指定要获取的参数字段。
     */
    @Retention(RetentionPolicy.RUNTIME)
    @Target({ElementType.PARAMETER})
    @Documented
    public @interface ParameterCacheKey {
        /**
         * 参数字段
         *
         * @return
         */
        String fieldName() default "";
    }
    
    Aspect
    @Service
    @Slf4j
    public class QueryCacheAspect {
    
        @Autowired
        private RedisTemplate redisTemplate;
    
    
        /**
         * 定义拦截规则:拦截所有@QueryCache注解的方法。
         */
        @Pointcut("@annotation(com.cnest.cacheredis.common.annotation.CacheKey)")
        public void queryCachePointcut() {
        }
    
        /**
         * 拦截器具体实现
         *
         * @param joinPoint
         * @return
         * @throws Throwable
         */
        @Around("queryCachePointcut()")
        public Object interceptor(ProceedingJoinPoint joinPoint) throws Throwable {
            long beginTime = System.currentTimeMillis();
            MethodSignature signature = (MethodSignature) joinPoint.getSignature();
            //获取被拦截的方法
            Method method = signature.getMethod();
            //获取方法注解
            CacheKey cacheKey = method.getAnnotation(CacheKey.class);
    
            CacheEvent cacheEvent = cacheKey.cacheEvent();
            String key = cacheKey.key();
    
            StringBuilder realKey = new StringBuilder(cacheKey.key());
    
            // 循环所有的参数
            Object[] args = joinPoint.getArgs();
            for (int i = 0; i < args.length; i++) {
                MethodParameter methodParam = new SynthesizingMethodParameter(method, i);
                //获取参数注解
                Annotation[] parameterAnnotations = methodParam.getParameterAnnotations();
                // 循环参数上所有的注解
                for (Annotation paramAnn : parameterAnnotations) {
                    if (paramAnn instanceof ParameterCacheKey) {
                        ParameterCacheKey parameterCacheKey = (ParameterCacheKey) paramAnn;
                        //取到ParameterCacheKey的标识参数的值
                        Object fieldValue = AopMethodUtil.getParamValue(args[i], parameterCacheKey.fieldName());;
                        realKey.append(":").append(parameterCacheKey.fieldName()).append(":").append(fieldValue);
                        break;
                    }
                }
            }
    
            //获取不到key值,抛异常
            if (StringUtils.isBlank(realKey.toString())) throw new RuntimeException("****缓存key值不存在****");
            // 如果没有 ParameterCacheKey 注解 默认md5(类名+方法+参数)
            if ((Objects.equals(realKey.toString(), key))) {
                realKey.append(":").append(getKey(signature, joinPoint.getArgs()));
            }
            log.debug("获取到缓存key值 {} ", realKey);
            boolean hasKey = redisTemplate.hasKey(realKey.toString());
            if (hasKey) {
                // 根据对应操作
                if (Objects.equals(cacheEvent, CacheEvent.QUERY)) {
                    // 缓存中获取到数据,直接返回。
                    Object object = redisTemplate.opsForValue().get(realKey.toString());
                    log.debug("缓存命中耗时:{}", (System.currentTimeMillis() - beginTime));
                    return object;
                }
                if (Objects.equals(cacheEvent, CacheEvent.DELETE)) {
                    // 删除缓存
                    redisTemplate.delete(realKey.toString());
                    log.debug("缓存删除key {} ", realKey);
                }
            }
    
            Object object = joinPoint.proceed();
            //设置缓存
            if (Objects.equals(cacheEvent, CacheEvent.QUERY)) {
                redisTemplate.opsForValue().set(realKey.toString(), object, cacheKey.expireTime(),
                        cacheKey.timeUnit());
            }
    
            log.debug("结束耗时:{}", (System.currentTimeMillis() - beginTime));
            return object;
        }
    
        /**
         * 默认md5(类名+方法+参数)
         *
         * @param methodSignature
         * @param args
         * @return
         */
        private String getKey(MethodSignature methodSignature, Object[] args) {
            StringBuilder key = new StringBuilder(methodSignature.getDeclaringTypeName());
            key.append(methodSignature.getMethod().getName());
            Object[] paramValues = args;
            String[] paramNames = (methodSignature).getParameterNames();
            for (int i = 0; i < paramNames.length; i++) {
                key.append(paramValues[i]);
            }
            return MD5.create().digestHex(key.toString());
        }
    }
    
    
    @Slf4j
    public class AopMethodUtil {
        /**
         * 基础数据类型
         */
        private static List<String> types =
                Arrays.asList("java.lang.Integer", "java.lang.Double",
                        "java.lang.Float", "java.lang.Long", "java.lang.Short",
                        "java.lang.Byte", "java.lang.Boolean", "java.lang.Char",
                        "java.lang.String","java.math.BigDecimal", "int", "double", "long", "short", "byte",
                        "boolean", "char", "float");
    
        /**
         * 获取参数值
         *
         * @param arg
         * @param fieldName 参数字段名
         * @return
         */
        public static Object getParamValue(Object arg, String fieldName) {
            Object value = "";
            // 获取对象类型
            if (Objects.nonNull(arg)) {
                String typeName = arg.getClass().getTypeName();
                //1 判断是否是基础类型
                if (types.contains(typeName)) {
                    value = arg;
                } else {
                    //2 通过反射获取实体类属性
                    value = getFieldsValue(arg, fieldName);
                }
            }
            return value;
        }
    
        /**
         * 解析实体类,获取实体类中的指定属性値
         *
         * @param object    参数实体对象
         * @param fieldName 字段
         * @return
         */
        public static Object getFieldsValue(Object object, String fieldName) {
            //通过反射获取所有的字段,getFileds()获取public的修饰的字段
            //getDeclaredFields获取private protected public修饰的字段
            Field[] fields = object.getClass().getDeclaredFields();
            String typeName = object.getClass().getTypeName();
            if (types.contains(typeName)) {
                return object;
            }
            Object value = "";
            for (Field field : fields) {
                //在反射时能访问私有变量
                field.setAccessible(true);
                try {
                    //如果实体类里面继续包含实体类,就没法获取。
                    //我们可以通递归的方式去处理实体类包含实体类的问题。
                    if (types.contains(field.getType().getName())) {
                        if (field.getName().equals(fieldName)) {
                            value = field.get(object) == null ? "" : field.get(object).toString();
                            return value;
                        }
                    }
                } catch (IllegalArgumentException | IllegalAccessException e) {
                    log.error("error {}",e);
                }
            }
            return value;
        }
    
        /**
         * 解析实体类,获取实体类中的属性
         *
         * @param object
         * @return
         */
        public static Object getFieldsValue(Object object) {
            //通过反射获取所有的字段,getFileds()获取public的修饰的字段
            //getDeclaredFields获取private protected public修饰的字段
            Field[] fields = object.getClass().getDeclaredFields();
            String typeName = object.getClass().getTypeName();
            if (types.contains(typeName)) {
                return object;
            }
            StringBuilder sb = new StringBuilder();
            sb.append("{");
            for (Field f : fields) {
                //在反射时能访问私有变量
                f.setAccessible(true);
                try {
                    //这边会有问题,如果实体类里面继续包含实体类,这边就没法获取。
                    //其实,我们可以通递归的方式去处理实体类包含实体类的问题。
                    if (types.contains(f.getType().getName())) {
                        sb.append(f.getName()).append(" : ").append(f.get(object)).append(", ");
                    }
                } catch (IllegalArgumentException | IllegalAccessException e) {
                    log.error("error {}",e);
                }
            }
            sb.append("}");
            return sb.toString();
        }
    }
    

    2.3 也可以支持 不过我没有用而已 就是 解析表达式SPEL

    2.4 demo 是不是很简单

     /**
      *  id删除
     */
        @ApiOperation(value = "id删除",notes = "通用结果返回对象")
        @PostMapping("user/delete")
        @CacheKey(cacheEvent = CacheEvent.DELETE)
        public CommonResult<Boolean> delete(@ParameterCacheKey(fieldName = "id") User user){
            Boolean success = userService.removeById(user.getId());
            return CommonResultResponse.ok(success);
        }
    
        /**
        *  id查询
        */
        @ApiOperation(value = "id查询",notes = "通用结果返回对象")
        @GetMapping("user/findById")
        @CacheKey
        public CommonResult<User> findById(@ParameterCacheKey(fieldName = "id") Integer id, HttpServletResponse response, HttpServletRequest request, MultipartFile file){
            User user = userService.getById(id);
            return CommonResultResponse.ok(user);
        }
    

优点 可以省略缓存判断操作 单一职责的方法

缺点 带有多重缓存业务逻辑操作 还得自己灵活编写

posted @ 2021-03-31 10:10  川流不息&  阅读(136)  评论(0编辑  收藏  举报