面向切面编程AOP例子

本文介绍了如何在Gmall应用中通过注解技术,结合RedisTemplate和RedissonClient实现分布式缓存和分布式锁的功能,以及如何在切点上添加注解以控制缓存操作和锁定策略。
摘要由CSDN通过智能技术生成

声明注解 模仿@Transactianal

@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @interface GmallCache {
    /**
     * 缓存key的前缀
     *
     * @return
     */
    String prefix() default "cache";

    /**
     * 锁的后缀
     *
     * @return
     */
    String suffix() default "cache";
}

添加切面类 纵向重复代码 横向提取出来 以分布式缓存+分布式锁代码为例

@Component
@Aspect
public class GmallCacheAspect {
    @Autowired
    private RedisTemplate redisTemplate;
    @Autowired
    private RedissonClient redissonClient;

    @SneakyThrows
    @Around("@annotation(com.atguigu.gmall.common.cache.GmallCache)")//切GmallCache注解
    public Object cacheAroundAdvice(ProceedingJoinPoint joinPoint) throws Throwable {
        //声明一个对象
        Object object = new Object();
        try {
            //  在环绕通知中处理业务逻辑 {实现分布式锁}
            MethodSignature signature = (MethodSignature) joinPoint.getSignature();
            //  获取到注解,注解使用在方法上!
            GmallCache gmallCacheAnnotation = signature.getMethod().getAnnotation(GmallCache.class);
            //获取key前缀 后缀
            String prefix = gmallCacheAnnotation.prefix();
            String suffix = gmallCacheAnnotation.suffix();
            //获取参数
            Object[] args = joinPoint.getArgs();
            //拼key
            String key = prefix + Arrays.asList(args) + suffix;
            //锁key
            String lockkey = prefix + Arrays.asList(args) + RedisConst.SKULOCK_SUFFIX;
            //判断缓存中是否存在该数据
//      下面类似  Object o = redisTemplate.opsForValue().get(key);
            object = cacheHit(key, signature);
            //判断是否返回数据
            if (null == object) {
                //获取锁
                RLock lock = redissonClient.getLock(lockkey);
                boolean flag = lock.tryLock(RedisConst.SKULOCK_EXPIRE_PX1, RedisConst.SKULOCK_EXPIRE_PX2, TimeUnit.SECONDS);
                try {
                    if (flag) {
                        //查数据库 执行方法体
                        object = joinPoint.proceed(joinPoint.getArgs());
                        if (null == object) {
                            //数据库为空
                            object = new Object();
                            //数据库不为空   放入缓存
                            Class returnType = signature.getReturnType();
                            object = returnType.newInstance();
                            redisTemplate.opsForValue().set(key, JSON.toJSONString(object), RedisConst.SKUKEY_TEMPORARY_TIMEOUT, TimeUnit.SECONDS);
                            return object;
                        }
                        redisTemplate.opsForValue().set(key, JSON.toJSONString(object), RedisConst.SKUKEY_TEMPORARY_TIMEOUT, TimeUnit.SECONDS);
                        return object;
                    } else {
                        Thread.sleep(1000);
                        this.cacheAroundAdvice(joinPoint);
                    }
                } finally {
                    lock.unlock();
                }

            } else {
                return object;
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
        //兜底方法
        return joinPoint.proceed(joinPoint.getArgs());
    }

    /**
     * 表示从缓存中获取数据
     *
     * @param key       缓存的key
     * @param signature 获取方法的返回值类型
     * @return
     */
    private Object cacheHit(String key, MethodSignature signature) {
        //透过key来获取缓存数据
        String strJson = (String) redisTemplate.opsForValue().get(key);
        //  表示从缓存中获取到了数据
        if (!StringUtils.isEmpty(strJson)) {
            //  字符串存储的数据是什么?   就是方法的返回值类型
            Class returnType = signature.getReturnType();
            //将字符串变为当前的返回值类型
            return JSON.parseObject(strJson,returnType);
        }
        return null;
    }
}

再切点上加注解 并指定前后缀

    @GmallCache(prefix = "skuValueIdsMap:", suffix = RedisConst.SKUKEY_SUFFIX)
    @Override
    public List<Map> getSkuValueIdsMap(Long spuId) {
        return skuSaleAttrValueMapper.selectSkuValueIdsMap(spuId);
    }

    @GmallCache(prefix = "spuSaleAttrListCheckBySku:", suffix = RedisConst.SKUKEY_SUFFIX)
    @Override
    public List<SpuSaleAttr> getSpuSaleAttrListCheckBySku(Long skuId, Long spuId) {
        return spuSaleAttrMapper.selectSpuSaleAttrListCheckBySku(skuId, spuId);
    }

  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值