redis之缓存的收益与成本、缓存更新策略、双写一致性、缓存粒度控制、缓存穿透,缓存击穿,缓存雪崩及Redis-Cluster

1、缓存的收益与成本

1.1、受益
1、加速读写
2、降低后端负载: 后端服务器通过前端缓存降低负载,业务端使用redis降低后端mysql负载
1.2、成本
1、数据不一致:缓存层和数据层有时间窗口不一致,和更新策略有关
2、代码维护成本: 多了一层缓存逻辑
3、运维成本: 比如使用了Redis Cluster
1.3、使用场景
1、降低后端负载: 对高消耗的sql,join结果集/分组统计的结果做缓存
2、加速请求响应: 利用redis优化io响应时间
3、大量写合并为批量写: 如计数器先redis累加再批量写入db

2、缓存更新策略

1、LRU/LFU/FIFO算法剔除: 例如maxmemory-policy(到了最大内存,对应的应对策略)(1) LRU -Least Recently Used   没有被使用时间最长的(保证热点数据)
  (2) LFU -Least Frequenty User  一定时间段内使用次数最少的
  (3) FIFO -First In First Out   先进先出

maxmemory-policy,超过最大内存,新的放不进去了,淘汰策略

LIRS (Low Inter-reference Recency Set)是一个页替换算法,相比于LRU(Least Recently Used)和很多其他的替换算法,LIRS具有较高的性能,这是通过使用两次访问同一页之间的距离(本距离指中间被访问了多少非重复块)作为一种尺度去动态地将访问页排序,从而去做一个替换的选择

如何保证redis中数据是最热的,配置lru的剔除算法 
配置文件中: maxmemory-policy:volatile-lru

配置文件中设置:

># LRU配置
>maxmemory-policy:volatile-lru
>(1) noeviction: 如果内存使用达到了maxmemory,client还要继续写入数据,那么就直接报错给客户端
>(2) allkeys-lru: 就是我们常说的LRU算法,移除掉最近最少使用的那些keys对应的数据,ps最长用的策略
>(3) volatile-lru: 也是采取LRU算法,但是仅仅针对那些设置了指定存活时间(TTL)的key才会清理掉
>(4) allkeys-random: 随机选择一些key来删除掉
>(5) volatile-random: 随机选择一些设置了TTL的key来删除掉
>(6) volatile-ttl: 移除掉部分keys,选择那些TTL时间比较短的keys

># LFU配置 Redis4.0之后为maxmemory_policy淘汰策略添加了两个LFU模式
>volatile-lfu: 对有过期时间的key采用LFU淘汰算法
>allkeys-lfu: 对全部key采用LFU淘汰算法
># 还有2个配置可以调整LFU算法:
>lfu-log-factor 10
>lfu-decay-time 1
># lfu-log-factor可以调整计数器counter的增长速度,lfu-log-factor越大,counter增长的越慢
># lfu-decay-time是一个以分钟为单位的数值,可以调整counter的减少速度

2、超时剔除: 例如expire,设置过期时间

3、主动更新: 开发控制生命周期
策略一致性维护成本
LRU/LIRS算法剔除最差
超时剔除较差
主动更新

1、低一致性:最大内存和淘汰策略
2、高一致性:超时剔除和主动更新结合,最大内存和淘汰策略兜底

3、双写一致性(redis和mysql数据同步方案)

1、先更新数据库,再更新缓存(一般不用)
2、先删缓存,再更新数据库(在存数据的时候,请求来了缓存不是最新的)
3、先更新数据库,再删缓存(推荐用)

4、缓存粒度控制

在这里插入图片描述

1、从mysql获取用户信息:select * from user where id=100
2、设置用户信息缓存: set user:100 select * from user where id=100
3、缓存粒度:(1) 缓存全部属性
​  (2) 缓存部分重要属性

1、通用性:全量属性更好
2、占用空间:部分属性更好
3、代码维护:表面上全量属性更好

5、缓存穿透,缓存击穿,缓存雪崩

# 缓存穿透
# 描述:
缓存穿透是指缓存和数据库中都没有的数据,而用户不断发起请求,如发起为id为“-1”的数据或id为特别大不存在的数据,这时的用户很可能是攻击者,攻击会导致数据库压力过大
# 解决方案:
1、接口层增加校验,如用户鉴权校验,id做基础校验,id<=0的直接拦截
2、从缓存取不到的数据,在数据库中也没有取到,这时也可以将key-value对写为key-null,缓存有效时间可以设置短点,30(设置太长会导致正常情况也没法使用),这样可以防止攻击用户反复用同一个id暴力攻击
3、通过布隆过滤器实现

# 缓存击穿
# 描述:
缓存击穿是指缓存中没有但数据库中有的数据(一般是缓存时间到期),这时由于并发用户特别多,同时读缓存没读到数据,又同时去数据库去取数据,引起数据库压力瞬间增大,造成过大压力
# 解决方案:
设置热点数据永远不过期
 
# 缓存雪崩
# 描述:
缓存雪崩是指缓存中数据大批量到过期时间,而查询数据量巨大,引起数据库压力过大甚至down机,和缓存击穿不同的是,缓存击穿指并发查同一条数据,缓存雪崩是不同数据都过期了,很多数据都查不到从而查数据库
# 解决方案:
1、缓存数据的过期时间设置随机,防止同一时间大量数据过期现象发生
2、如果缓存数据库是分布式部署,将热点数据均匀分布在不同搞得缓存数据库中
3、设置热点数据永远不过期

6、Redis-Cluster

6.1、Redis Cluser背景
1、并发量: 单机redis qps为10w/s,但是我们可能需要百万级别的并发量
2、数据量: 机器内存16g-256g,如果存500g数据呢?

解决: 加机器,分布式
redis cluster在2015年的3.0版本加入了,满足分布式的需求
6.2、配置部署
cd /opt/soft/redis/config
vim redis-7000.conf
# 写入
port 7000
daemonize yes
dir "/opt/soft/redis/data/"
logfile "7000.log"
dbfilename "dump-7000.rdb"
cluster-enabled yes                       # 开启cluster
cluster-config-file nodes-7000.conf       # 给cluster节点增加一个自己的配置文件
cluster-require-full-coverage yes         # 只要集群中有一个故障了,整个就不对外提供服务了,这个实际不合理,假设有50个节点,一个节点故障了,所有不提供服务了,需要设置成no

# 快速生成其他配置
sed 's/7000/7001/g' redis-7000.conf > redis-7001.conf
sed 's/7000/7002/g' redis-7000.conf > redis-7002.conf
sed 's/7000/7003/g' redis-7000.conf > redis-7003.conf
sed 's/7000/7004/g' redis-7000.conf > redis-7004.conf
sed 's/7000/7005/g' redis-7000.conf > redis-7005.conf

# 启动
redis-server ./config/redis-7000.conf
redis-server ./config/redis-7001.conf
redis-server ./config/redis-7002.conf
redis-server ./config/redis-7003.conf
redis-server ./config/redis-7004.conf
redis-server ./config/redis-7005.conf

# 表示给每个主节点配置一个从节点
redis-cli --cluster create --cluster-replicas 1 127.0.0.1:7000 127.0.0.1:7001 127.0.0.1:7002 127.0.0.1:7003 127.0.0.1:7004 127.0.0.1:7005

# 登录
redis-cli -p 7000
CLUSTER NODES  # 集群节点信息
cluster slots  # 查看槽的信息
6.3、集群伸缩
6.3.1、伸缩原理
加入节点,删除节点: 槽和数据在节点之间的移动

在这里插入图片描述

6.3.2、集群扩容
# 作用: 为它迁移槽和数据实现扩容,作为从节点负责故障转移

# 1、准备新节点
集群模式
配置和其他节点统一
启动后是孤儿节点
sed 's/7000/7006/g' redis-7000.conf > redis-7006.conf
sed 's/7000/7007/g' redis-7000.conf > redis-7007.conf
redis-server conf/redis-7006.conf
redis-server conf/redis-7007.conf
# 孤立状态re
redis-cli -p 7006 cluster nodes

# 2、加入集群
# 方式一7000上执行
redis-cli -p 7000 cluster meet 127.0.0.1 7006
redis-cli -p 7000 cluster meet 127.0.0.1 7007

# 方式二
redis-cli --cluster add-node 127.0.0.1:7006 127.0.0.1:7000
redis-cli --cluster add-node 127.0.0.1:7007 127.0.0.1:7000

# 查看配置
cluster nodes
# 把7007做为7006的从
redis-cli -p 7007 cluster replicate 7006id
# 3、迁移槽和数据
# 槽迁移计划
# 迁移数据
# 添加从节点
  
# 我们不操作原生,直接使用redis-trip
redis-cli --cluster reshard 127.0.0.1:7000 
# 打印当前集群状态
# 希望迁移多少个槽: 4096
# 希望那个id是接收的: 7006的id
# 传入source id : all
# yes

# 查看
redis-cli -p 7000 cluster nodes
redis-cli -p 7000 cluster slots
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值