SpringBoot 基于自定义注解的Redis AOP操作

使用Spring的@Cacheable实现Reids缓存时,会遇到过期时间不能直接在注解中赋值的问题,在项目中自己动手实现一个基于注解的RedisCache处理,可以在注解中直接指定Key(键)和Expired(过期时间)。然后AOP拦截,解析注解并做相应处理。

注解定义如下:

@Documented
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface RedisCache {

    /**
     * 键
     * @return
     */
    String key() default "";

    /**
     * 过期时间
     * @return
     */
    long expired() default -1;

    /**
     * 是否为查询操作
     * 如果为写入数据库的操作,该值需置为 false
     * @return
     */
    boolean read() default true;
}

AOP拦截类如下:

@Component
@Aspect
public class RedisCacheAspect {
    private final static Logger logger = LoggerFactory.getLogger(RedisCacheAspect.class);

    @Autowired
    private RedisTemplate redisTemplate;

    /**
     * 定义切入点,使用了 @RedisCache 的方法
     */
    @Pointcut("@annotation(RedisCache)")
    public void redisServicePoint(){

    }

    /**
     * 环绕通知,方法拦截器
     */
    @Around("redisServicePoint()")
    public Object WriteReadFromRedis(ProceedingJoinPoint point){
        logger.info("<====== 进入 redisCache 环绕通知 ======>");
        try {
            // 获取RedisCache注解
            RedisCache redisCache = ((MethodSignature)point.getSignature()).getMethod().getAnnotation(RedisCache.class);
            if(redisCache != null && redisCache.read()){
                // 查询操作
                Object obj = redisTemplate.opsForValue().get(redisCache.key());
                if(obj == null){
                    // Redis 中不存在,则从数据库中查找,并保存到 Redis
                    logger.info("<====== Redis 中不存在该记录,从数据库查找 ======>");
                    obj = point.proceed();
                    if(obj != null) {
                        if(redisCache.expired() > 0) {
                            redisTemplate.opsForValue().set(redisCache.key(), obj, redisCache.expired(), TimeUnit.SECONDS);
                        }else {
                            redisTemplate.opsForValue().set(redisCache.key(), obj);
                        }
                    }
                }

                return obj;
            }
        }catch (Throwable ex){
            logger.error("<====== RedisCache 执行异常: {} ======>", ex);
        }

        return null;
    }
}

在查询方法中使用注解:

@RedisCache(key = "test_user", expired = 30000)
public User findById(Long uid) {
    return userMapper.selectByPrimaryKey(uid);
}

这样就实现了一个简单基于AOP的Redis缓存处理过程,该方法可以根据需要进行各种扩展。


好的,关于SpringBoot自定义Redis注解AOP的问题,我可以为您提供一些基本的介绍和示例代码。 Redis作为一种高性能的缓存和数据存储解决方案,被广泛应用于各种应用程序中。在SpringBoot应用程序中,使用Redis通常需要编写大量的重复代码,比如获取Redis连接、执行Redis命令、释放Redis连接等。这些重复代码不仅增加了开发和维护的难度,还影响了应用程序的性能。而AOP作为一种切面编程的技术,可以很好地解决这些问题。 下面是一个简单的示例代码,演示如何通过自定义注解实现对Redis操作AOP处理: 首先,定义一个自定义注解: ```java @Target(ElementType.METHOD) @Retention(RetentionPolicy.RUNTIME) public @interface RedisCacheable { String key() default ""; long expire() default 0; } ``` 然后,在需要被拦截的方法上添加该注解: ```java @Component public class RedisService { @Autowired private RedisTemplate<String, String> redisTemplate; @RedisCacheable(key = "myKey", expire = 60) public String getValue() { return redisTemplate.opsForValue().get("myKey"); } } ``` 接下来,使用AspectJ的@Aspect注解定义一个切面类,并在该类中定义一个切点,用于匹配被@RedisCacheable注解的方法: ```java @Aspect @Component public class RedisAspect { @Autowired private RedisTemplate<String, String> redisTemplate; @Pointcut("@annotation(com.example.demo.annotation.RedisCacheable)") public void redisCacheablePointcut() {} @Around("redisCacheablePointcut()") public Object aroundRedisCacheable(ProceedingJoinPoint joinPoint) throws Throwable { MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature(); Method method = methodSignature.getMethod(); RedisCacheable redisCacheable = method.getAnnotation(RedisCacheable.class); String key = redisCacheable.key(); long expire = redisCacheable.expire(); String value = redisTemplate.opsForValue().get(key); if (value != null) { return value; } Object result = joinPoint.proceed(); if (result != null) { redisTemplate.opsForValue().set(key, result.toString()); if (expire > 0) { redisTemplate.expire(key, expire, TimeUnit.SECONDS); } } return result; } } ``` 在该切面类中,使用@Around注解定义一个环绕通知,在该通知中,首先获取被拦截方法上的@RedisCacheable注解,然后根据注解中的key值从Redis中获取数据。如果Redis中已经存在该数据,则直接返回;否则,执行被拦截方法,并将结果存储到Redis缓存中。 最后,启动SpringBoot应用程序,调用RedisService的getValue方法,就可以看到输出结果: ```java // 第一次调用,从数据库中获取数据,并将数据存入Redis缓存中 getValue... // 第二次调用,直接从Redis中获取数据 getValue... ``` 以上就是一个简单的SpringBoot自定义Redis注解AOP的示例。通过使用自定义注解AOP技术,可以更加方便地实现对Redis缓存的操作,并提高应用程序的性能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值