Redis缓存实战(2)

本文涉及的产品
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
云数据库 Tair(兼容Redis),内存型 2GB
简介: Redis缓存实战(2)

缓存定义

缓存(Cache)是数据交换的缓冲区,是存储数据的临时地方,读写性能较高。缓存的作用:降低后端负载并且提高读写时间,降低响应时间。缓存的成本:数据一致性成本,代码维护成本和运维成本。

Redis缓存实战


此代码模拟商户查询,查询商户时首先从redis缓存中查询是否有符合自己id的商户,如果有直接返回,如果没有然后从数据库中查询,如果数据库中查询到对应的数据,然后将该数据添加到redis缓存中,并且返回该数据,如果还是没有查询到就会返回异常信息。代码如下

   @GetMapping("/select")
    public Result select(Long id){
        TbShop tbShop = (TbShop) redisTemplate.opsForValue().get("shopcache" + id);
        if(tbShop!=null){
            return Result.success(tbShop);
        }
        TbShop tbShop1 = tbShopMapper.selectById(id);
        if(tbShop1==null){
            return Result.error("404","未发现该商铺");
        }
        redisTemplate.opsForValue().set("shopcache"+id,tbShop1,30,TimeUnit.MINUTES);
        return Result.success(tbShop1);
    }


对于更新数据库信息,缓存更新策略如下图

这里采用主动更新策略,在更新数据库的时候同时更新缓存。操作缓存和数据库有三个问题需要考虑如下。

1删除缓存还是更新缓存?

更新缓存:每次更新数据库都更新缓存,无效操作比较多

删除缓存:更新数据库让缓存失效,查询时更新缓存,效果比较好

2如何保证缓存与数据库的操作同时成功或者失败?

单体系统:将缓存和数据库放在一个事务中。

分布式系统:利用TCC等分布式事务方案。

3先操作数据库还是缓存?

先删除数据库再操作数据库会遇到如下问题

f12e94531f994e13960a47d813615398.png

经过如图所示过程就会出现缓存保存旧的数据的情况。并且这种情况的发生概念是非常高的。


先操作数据库,再删除缓存会出现如下情况


75231944090542ee955578a7871ab357.png


这种情况下也会发生缓存中存入旧数据的情况,但是由于缓存的执行时间远远小于数据库操作数据,所有这种情况的发生概念会比较小,因此可以选择先更新数据库,然后再删除缓存。


代码如下

   @Transactional
    @PostMapping("/update")
    public Result update(@RequestBody  TbShop tbShop){
        int update = tbShopMapper.updateById(tbShop);
        redisTemplate.delete("shopcache"+tbShop.getId());
        return Result.success();
    }

缓存问题

缓存穿透

缓存穿透是指客户端请求的数据在缓存和数据库中都不存在,这样缓存永远都不会生效,这些请求都会打到数据中。

常见的解决方案:缓存空对象或者布隆过滤。

缓存空对象优点:实现简单,维护方便,缺点:额外的内存消耗,可能造成短期的不一致。

布隆过滤:优点:内存占用少,没有多余的key。缺点:实现复杂和存在误判可能。

 

缓存雪崩

缓存雪崩是指在同一时段大量的缓存key同时失效或者Redis服务宕机,导致大量请求到达数据库,带来巨大压力。

解决方案

给不同的Key的TTL添加随机值

利用Redis集群提高服务的可用性

给缓存业务添加降级限流策略

给业务添加多级缓存

 

缓存击穿

缓存击穿问题也叫热点Key问题,就是一个被高并发访问并且缓存重建业务较复杂的key突然失效了,无数的请求访问会在瞬间给数据库带来巨大的冲击。

解决方案:可以通过互斥锁和逻辑过期的方式进行解决。

fc7949cff5684ef1b5a46a0c951ecd07.png


这里利用互斥锁的方式来进行模拟

互斥锁流程如下图

代码如下

 @GetMapping("/select1")
    public Result select1(Long id){
        TbShop tbShop = (TbShop) redisTemplate.opsForValue().get("shopcache" + id);
        if(tbShop!=null){
            return Result.success(tbShop);
        }
        TbShop tbShop1 = null;
        String lockkey = "lockshop"+id;
        try {
            boolean trylock = trylock(lockkey);
            if(!trylock){
                Thread.sleep(50);
               return select(id);
            }
            tbShop1 = tbShopMapper.selectById(id);
            System.out.println("数据库查询");
            Thread.sleep(200);
            if(tbShop1==null){
                return Result.error("404","未发现该商铺");
            }
            redisTemplate.opsForValue().set("shopcache"+id,tbShop1,30,TimeUnit.MINUTES);
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            unlock(lockkey);
        }
        return Result.success(tbShop1);
    }
  private boolean trylock(String key){
        Boolean aBoolean = redisTemplate.opsForValue().setIfAbsent(key, "1", 10, 
        TimeUnit.SECONDS);
        return BooleanUtil.isTrue(aBoolean);
    }
    private void unlock(String key){
        redisTemplate.delete(key);
    }

通过jmeter压测工具使用一千多个线程进行测试,后端仅仅只是执行了一次调用数据库操作,实验结果因此得到验证。

相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore     ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库 ECS 实例和一台目标数据库 RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
相关文章
|
5天前
|
缓存 NoSQL 关系型数据库
大厂面试高频:如何解决Redis缓存雪崩、缓存穿透、缓存并发等5大难题
本文详解缓存雪崩、缓存穿透、缓存并发及缓存预热等问题,提供高可用解决方案,帮助你在大厂面试和实际工作中应对这些常见并发场景。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:如何解决Redis缓存雪崩、缓存穿透、缓存并发等5大难题
|
6天前
|
存储 缓存 NoSQL
【赵渝强老师】基于Redis的旁路缓存架构
本文介绍了引入缓存后的系统架构,通过缓存可以提升访问性能、降低网络拥堵、减轻服务负载和增强可扩展性。文中提供了相关图片和视频讲解,并讨论了数据库读写分离、分库分表等方法来减轻数据库压力。同时,文章也指出了缓存可能带来的复杂度增加、成本提高和数据一致性问题。
【赵渝强老师】基于Redis的旁路缓存架构
|
14天前
|
缓存 NoSQL Redis
Redis 缓存使用的实践
《Redis缓存最佳实践指南》涵盖缓存更新策略、缓存击穿防护、大key处理和性能优化。包括Cache Aside Pattern、Write Through、分布式锁、大key拆分和批量操作等技术,帮助你在项目中高效使用Redis缓存。
84 22
|
13天前
|
缓存 NoSQL 中间件
redis高并发缓存中间件总结!
本文档详细介绍了高并发缓存中间件Redis的原理、高级操作及其在电商架构中的应用。通过阿里云的角度,分析了Redis与架构的关系,并展示了无Redis和使用Redis缓存的架构图。文档还涵盖了Redis的基本特性、应用场景、安装部署步骤、配置文件详解、启动和关闭方法、systemctl管理脚本的生成以及日志警告处理等内容。适合初学者和有一定经验的技术人员参考学习。
97 7
|
17天前
|
存储 缓存 监控
利用 Redis 缓存特性避免缓存穿透的策略与方法
【10月更文挑战第23天】通过以上对利用 Redis 缓存特性避免缓存穿透的详细阐述,我们对这一策略有了更深入的理解。在实际应用中,我们需要根据具体情况灵活运用这些方法,并结合其他技术手段,共同保障系统的稳定和高效运行。同时,要不断关注 Redis 缓存特性的发展和变化,及时调整策略,以应对不断出现的新挑战。
51 10
|
存储 缓存 NoSQL
Redis实战之入门进阶到精通
Redis 是一个远程内存数据库,它不仅性能强劲,而且还具有复制特性以及为解决问题而生的独一无二的数据模型。Redis 提供了 5 种不同类型的数据结构,各式各样的问题都可以很自然地映射到这些数据结构上:Redis 的数据结构致力于帮助用户解决问题,而不会像其他数据库那样,要求用户扭曲问题来适应数据库。除此之外,通过复制、持久化(persistence)和客户端分片(client-side sharding)等特性,用户可以很方便地将 Redis 扩展成一个能够包含数百 GB 数据、每秒处理上百万次请求的系统。
Redis实战之入门进阶到精通
|
1月前
|
存储 缓存 NoSQL
数据的存储--Redis缓存存储(一)
数据的存储--Redis缓存存储(一)
|
存储 NoSQL Java
当Java遇到Redis:Jedis实战入门
Redis是一个开源,高级的键值存储和一个适用的解决方案,用于构建高性能,可扩展的Web应用程序。本文将概要介绍Redis的特性和语法,并以实例代码的形式介绍如何通过Jedis在java语言环境下控制Redis,帮助各位读者快速入门。
1623 0

热门文章

最新文章