个人技术分享

代码如下:
public boolean checkCarInBlackListByCache(String carNumber, String projectCode, Long parkingAreaId) {
    Object o = redisUtil.get(getCarNumberCacheKey(carNumber, projectCode));
    if (Objects.isNull(o)) {
        ParkingBlackListQuery query = new ParkingBlackListQuery();
        query.setCarNumber(carNumber);
        query.setProjectCode(projectCode);
        List<ParkingBlackListBO> list = list(query);
        if (CheckUtils.isEmpty(list)) {
            return false;
        }
        ParkingBlackListBO parkingBlackListBO = list.get(0);
        redisUtil.set(getCarNumberCacheKey(carNumber, projectCode), parkingBlackListBO.getId());
        Date date = new Date();
        boolean valid = date.after(parkingBlackListBO.getEffectiveTimeBegin()) && date.before(parkingBlackListBO.getEffectiveTimeEnd());
        return valid && parkingBlackListBO.getParkingArea().contains(parkingAreaId.toString());
    }
    Long id = (Long) o;
    ParkingBlackListBO parkingBlackListBO = get(id);
    if (Objects.isNull(parkingBlackListBO)) {
        return false;
    }
    Date date = new Date();
    boolean valid = date.after(parkingBlackListBO.getEffectiveTimeBegin()) && date.before(parkingBlackListBO.getEffectiveTimeEnd());
    return valid && parkingBlackListBO.getParkingArea().contains(parkingAreaId.toString());
}
@Override
@Transactional(readOnly = true, rollbackFor = Throwable.class)
@Cacheable(value = "ParkingBlackListBOCache", key = "#id", unless = "#result == null")
public ParkingBlackListBO get(Long id) {
    if (id == null) {
        throw new RuntimeException("参数id不能为空");
    }
    // 查询数据之前的业务扩展
    getBefore(id);
    ParkingBlackListBO bo;
    try {
        // 执行数据库查询操作
        bo = getDao(id);
    } catch (Exception e) {
        throw new RuntimeException("持久化操作异常", e);
    }
    // 数据查询之后的扩展
    getAfter(bo);
    // 数据查询成功之后的扩展
    return getSuccess(bo);
}

当调用checkCarInBlackListByCache 该方法时,根据日志,发现每次还调用数据库查询。然后分析具体日志,发现调用get方法时,还是调用了数据库。判断是缓存注解没有起作用。

通过debug,跟踪源代码知道调用了缓存拦截器的方法org.springframework.cache.interceptor.CacheInterceptor#invoke

然后调用,切面注解支持方法进行具体的缓存注解处理org.springframework.cache.interceptor.CacheAspectSupport#execute(org.springframework.cache.interceptor.CacheOperationInvoker, java.lang.Object, java.lang.reflect.Method, java.lang.Object[])

从这里看出,是通过调用代理对象进行增强处理的。

而在当在checkCarInBlackListByCache 方法里面调用get方法时,依然是旧对象,还不是代理对象,因此需要重新注入新的代理对象调用。才能保证get方法可以获取到缓存的数据。最终通过引入新的注入对象,进行调用

ParkingBlackListBO parkingBlackListBO = parkingBlackListService.get(id);

问题得到解决