Spring @Cacheable注解类内部调用不生效

在使用Spring Cache时遇到一个问题,@Cacheable注解的removeGroupUserStatusCached方法在内部调用时不生效,导致缓存不一致。原因是内部调用不经过动态代理,无法触发缓存逻辑。解决办法包括将方法移到单独类,通过代理类调用,或利用AopContext获取当前代理对象执行方法。

最近会员中心拼团中一个查询接口做了缓存,使用到spring cache框架。在用户状态变更的时候,清除用户的缓存cache。但是实际结果是数据库状态变了,但是缓存状态并没有清除,

结果出现缓存不一致的问题。具体如下:

 @CacheEvict(value = "groupUserCached", key = "'user_status_' + #userId")
    public void removeGroupUserStatusCached(long userId) {
        logger.info("removeGroupUserStatusCached, userId is {}", userId);
    }

    /**
     * 支付成功新客变老客
     * @param userId
     */
    public void updateUserStatusToOld(Long userId){
        GroupUserDO groupUserDO = new GroupUserDO();
        groupUserDO.setUserId(userId);
        groupUserDO.setStatus(GroupUserStatus.OLD_USER.getStatus());
        int result = groupUserDao.updateStatusByUserId(groupUserDO);
        if(result > 0){
            this.unLockGroupUser(userId);
            this.removeGroupUserStatusCached(userId);
            logger.info("【新客变老客】会员状态更新成功, userId:{}", userId);
        }else{
            logger.info("【新客变老客】会员状态更新失败, userId:{}", userId);
        }
    }

方法updateUserStatusToOld调用了注解方法removeGroupUserStatusCached,但是方法removeGroupUserStatusCached没有生效。
要解决这个问题,首先我们需要了解spring缓存的原理。spring cache的实现原理跟spring的事物管理类似,都是基于AOP的动态代理实现的:即都在方法调用前后
去获取方法的名称、参数、返回值,然后根据方法名称、参数生成缓存的key(自定义的key例外),进行缓存。debug可以看到调用的实际上是groupUserBiz
的代理类,基于cglib动态代理实现。当调用代理的方法时,代理可以整体控制实际的方法的入参和返回值。比如缓存结果,直接跳过执行实际的方法等。
这里写图片描述

下面这个是内部直接调用注解方法,如果是在同一个类的内部直接调用的话,调用的是groupUserBiz的实例,不会走代理。因此,注解缓存不会生效。
这里写图片描述

回到上面那个问题,removeGroupUserStatusCached()属于内部方法,直接调用removeGroupUserStatusCached()时,是不走代理的。

解决方法

1.可以把removeGroupUserStatusCached方法单独抽取到另外一个类里面,通过代理类引入。
2.通过((GroupUserService)AopContext.currentProxy()).removeGroupUserStatusCached方法获取当前类的代理类;
3.通过ApplicationContext获取当前类的代理对象,GroupUserService groupUserService = springContextUtil.getBean(GroupUserService.class);

评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值