redis实战(商品查询缓存)

根据id查询商品:

添加redis缓存(增强读写功能)步骤

1.从redis中获取商品信息,判断信息是否存在

2.若存在则从redis中获取,不存在则直接查询数据库

3.若数据库中不存在则返回错误信息

4.若数据库中存在则同同样将信息存入redis中

controller层

复制代码
    @GetMapping("/{id}")
    public Result queryShopById(@PathVariable("id") Long id) {
        return shopService.queryById(id);
    }

service层

复制代码
@Resource
    private StringRedisTemplate stringRedisTemplate;

    @Override
    public Result queryById(Long id) {
        String key=CACHE_SHOP_KEY+id;
        //从redis查询商品缓存
        String shopJson = stringRedisTemplate.opsForValue().get(key);

        //判断是否命中,若命中
        if(StrUtil.isNotBlank(shopJson)){
            Shop shop = JSONUtil.toBean(shopJson, Shop.class);
            return Result.ok(shop);
        }
        //判断命中是否为空值
        if(shopJson!=null){
            //返回一个错误信息
            return Result.fail("店铺不存在");
        }

        //若不存在,查询数据库
        Shop shop = getById(id);

        if(shop==null){
            //将空值写入redis
            stringRedisTemplate.opsForValue().set(key,"",CACHE_NULL_TTL,TimeUnit.MINUTES);
            return Result.fail("店铺不存在");
        }
        //存入redis
        stringRedisTemplate.opsForValue().set(key,JSONUtil.toJsonStr(shop),CACHE_SHOP_TTL, TimeUnit.MINUTES);

        return Result.ok(shop);
    }

有可能发生数据库和redis中信息存储不一致的情况,即缓存更新问题

对于低一致性的需求:可以使用内存淘汰,例如商品店铺的查询缓存

对于高一致性的需求:使用主动更新,并以超时剔除为兜底方案,例如商品详情查询的缓存

然后应该先操作数据库,再删除缓存

查询店铺

controller类

复制代码
    @GetMapping("/{id}")
    public Result queryShopById(@PathVariable("id") Long id) {
        return shopService.queryById(id);
    }

service类

复制代码
    @Resource
    private StringRedisTemplate stringRedisTemplate;

    @Override
    public Result queryTypeList() {
        String key=CACHE_SHOP_TYPE_KEY;
        //从redis中读取缓存
        String type = stringRedisTemplate.opsForValue().get(key);

        //判断是否命中缓存
        if(StrUtil.isNotBlank(type)){
            //若存在
            List<ShopType> list = JSONUtil.toList(type, ShopType.class);
            return Result.ok(list);
        }

        //若不存在,查找数据库
        List<ShopType> shopTypes = query().orderByAsc("sort").list();
        if(shopTypes==null){
            return Result.fail("分类不存在!");
        }

        //存入redis
        stringRedisTemplate.opsForValue().set(key,JSONUtil.toJsonStr(shopTypes));

        return Result.ok(shopTypes);
    }

还需考虑缓存穿透问题

解决方案

1.从redis中获取信息,判断信息是否存在(再判断是否为空值)

2.若存在则从redis中获取,不存在则直接查询数据库

3.若数据库中不存在则设置为空值存入redis并设置有效时长

4.若数据库中存在则同同样将信息存入redis中

缓存雪崩

相关推荐
014-code5 小时前
订单超时取消与库存回滚的完整实现(延迟任务 + 状态机)
java·开发语言
java1234_小锋6 小时前
Java高频面试题:Springboot的自动配置原理?
java·spring boot·面试
末央&7 小时前
【天机论坛】项目环境搭建和数据库设计
java·数据库
枫叶落雨2227 小时前
ShardingSphere 介绍
java
花花鱼7 小时前
Spring Security 与 Spring MVC
java·spring·mvc
言慢行善8 小时前
sqlserver模糊查询问题
java·数据库·sqlserver
专吃海绵宝宝菠萝屋的派大星8 小时前
使用Dify对接自己开发的mcp
java·服务器·前端
大数据新鸟8 小时前
操作系统之虚拟内存
java·服务器·网络
Tong Z8 小时前
常见的限流算法和实现原理
java·开发语言
凭君语未可8 小时前
Java 中的实现类是什么
java·开发语言