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

最近会员中心拼团中一个查询接口做了缓存,使用到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动态代理实现。当调用代理的方法时,代理可以整体控制实际的方法的入参和返回值。比如缓存结果,直接跳过执行实际的方法等。
Spring @Cacheable注解类内部调用不生效_第1张图片

下面这个是内部直接调用注解方法,如果是在同一个类的内部直接调用的话,调用的是groupUserBiz的实例,不会走代理。因此,注解缓存不会生效。
Spring @Cacheable注解类内部调用不生效_第2张图片

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

解决方法

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

你可能感兴趣的:(Spring)