Redis简介
Redis特性
- 速度快(内存,10w QPS, C , 单线程)
- 持久化(将内存数据异步更新到磁盘,RDB&AOF)
- 多种数据结构(string list set zset hash BitMaps HyperLogLog GEO)
- 支持多语言
- 功能丰富(发布订阅 事务 Lua脚本 pipeline)
- 简单(23000 lines of code 不依赖外部库 单线程模型)
- 主从复制
单线程为什么这么快?
- 纯内存
- 非阻塞IO,使用多路复用机制,提升连接并发度。
- 避免多线程切换和竞态消耗
Redis典型应用场景
- 缓存系统(缓存)
- 计数器(微博转发和评论)
- 消息队列系统
- 排行榜
- 社交网络
- 实时系统(垃圾邮件–布隆过滤器)
Redis可执行文件说明
redis-server ---> Redis服务器
redis-cli ---> Redis客户端
redis-benchmark ---> Redis性能测试
redis-check-aof ---> AOF文件修复
redis-check-dump ---> RDB文件检查
redis-sentinel ---> Sentinel服务器
Redis 通用API
keys ---> 遍历所有key,热备从节点 O(n)
dbsize ---> 计算key的总数 O(1)
exist key ---> 检查key是否存在,返回0或1 O(1)
del key ---> 删除指定的key O(1)
expire key seconds ---> key在seconds秒后过期 O(1)
ttl key ---> 查看key的剩余过期时间 O(1)
persist key ---> 去掉key的过期时间 O(1)
type key ---> 查看当前key的类型 O(1)
常见数据结构
String应用场景:字符串缓存、计数器、分布式锁
常用API:
get key
del key
incr 整型自增
decr 整型自减
incrby key k key自增k,如果key不存在,自增后get(key) = k
decr key k key自减k,如果key不存在,自减后get(key) = k
set key value 无论是否存在
setnx key value key不存在才能设置
set key value xx key存在,才设置
mget key1 key2 批量获取
mset key1 value1 key2 value2 批量设置
getset key newvalue set key newvalue并返回旧的value
append key value 将value追加到旧的value中
strlen key 返回字符串长度
incrbyfloat key 3.5 增加key对应的值3.5
getrange key start end 获取字符串指定下标所有的值
setrange key index value 设置指定下标所有对应的值
Hash应用场景:存储键值对数据
常用API:
hget key field 获取hash key对应的field的value
hset key field value 设置hash key对应field的value
hdel key field 删除
hexists key field 判断hash key是否有field
hlen key 获取hash key field长度
hmget key field1 field2 批量获取
hmset key1 field1 value1 批量设置
hgetall key 返回hash key对应的所有的field和value
hvals key 返回hash key对应所有的field的value
hkeys key 返回hash key对应的所有field
hsetnx key field value 设置hash key对应的field的value
hincrby key field intCounter hash key 对应的field的value自增intCounter
hincrbyfloat
List特点:可重复的、可左右弹出压入的
常用API:
rpush key value1 value2 从列表右端进行插入
lpush
linsert key before|after value newValue 在list指定的前后newValue
lpop key 左边弹出一个元素
rpop key 右边弹出一个元素
lrem key count value 根据count值,删除所有value相等的项,count>0 从左往右删除。count<0从右往左,count=0删除所有value相等的值
ltrim key start end 按照索引范围修剪列表
lrange key start end 获取列表指定所有范围的item
lindex key index 获取列表指定索引的item
llen key 获取列表长度
lset key index newValue 设置指定索引为newValue
blpop key timeout lpop阻塞版本 timeout是阻塞超时时间
brpop rpop阻塞版本
Set特点:无序、不重复
常用API:
sadd key element 向集合key中添加element
srem key element 将集合中的element移出
scard key 计算集合大小
sismember key element 判断是否在集合中
srandmember key count 从集合中随机挑出count个元素
smembers key 获取集合所有元素
spop key 随机弹出一个元素
sdiff key1 key2 差集
sinter key1 key2 交集
sunion key1 key2 并集
zset特点:有序、无重复元素
常用API:
zadd key score element 添加score和element O(logN)
zrem key element 删除元素 O(1)
zscore key element 返回元素的分数
zincrby key increScore element 增加或者减少元素的分数
zcard key 返回元素的总个数
zrank
zrange key start end 返回索引内的元素
zrangebyscore key minScore maxScore 返回指定分数范围内的升序元素
zcount key minScore maxScore 返回有序集合内指定范围内的个数
zremrangebyrank key start end 删除指定排名内的升序元素
zremrangebyscore key min max 删除指定分数范围内的升序元素
zrevrank
zrevrange
zrevrangebyscore
zinterstore
zunionstore
Bitmap
位图:字符以二进制格式存储在内存中
常用API:
setbit key offset value 给位图指定索引设置值
getbit key offset
bitcount key [start end] 获取位值为1的个数
bitop op destkey key 做多个bitmap的and or not xor操作并将结果保存到destKey中
bitpos key targetBit [start] [end] 计算位图指定范围第一个偏移量对应的值等于targetBit的位置
HyperLogLog
特点如下:
- 基于HyperLogLog算法:极小空间完成独立数量统计
- 本质还是string
常用API:
pfadd key element 添加元素
pfcount key 计算独立总数
pfmerge destKey sourcekey [sourcekey] 合并多个hyperloglog
GEO
GEO:存储经纬度,计算两地距离,范围计算
常用API:
geoadd key longitude latitude member 增加地理位置信息
geoadd cities:locations 116.28 39.55 beijing
geopos key member 获取地理位置信息
geodist key member1 member2 [unit] 获取两地位置的距离,unit:m,km,mi,ft
georadius
- since 3.2 +
- type geoKey = zset
- 没有删除API:zrem key member
其他功能
慢查询
慢查询日志就是系统在命令执行前后计算每条命令的执行时间,当超过预设阈值,就将这条命令的相关信息记录下来,以便后期的问题排查和调优。
默认配置参数
slowlog-max-len = 128 # 最多能保存的慢查询日志条数,若超过采用FIFO方案将最旧的慢查询日志先删除。
slowlog-log-slower-than = 10000 # 慢查询的预设阀值(单位:微秒)
如果slowlog-log-slower-than的值是0,则会记录所有命令。
如果slowlog-log-slower-than的值小于0,则任何命令都不会记录日志。
慢查询命令
slowlog get [n] 获取慢查询队列
slowlog len 获取慢查询队列长度
slowlog reset 清空慢查询队列
Pipeline
客户端将多个请求打包,同时发给服务端。比较是否非实时需求,统一打包一次发送,减少网络传输的开销和时延问题。
事务
事务将一种或多种请求打包,然后一次性按顺序 执行多个命令的机制,事务执行过程中,服务器不会中断事务去执行其他请求。事务以MULTI
命令开始,接着将多个命令放入事务中,由EXEC
将这个事务提交给服务器。
发布订阅模式
主要有以下角色:
- 发布者(publisher)
- 订阅者(subscriber)
- 频道(channel)
常用API
publish channel message
subscribe channel
unsubscribe channel
psubscribe [pattern] 订阅模式
punsubscribe [pattern] 退出指定的模式
pubsub channels 列出至少有一个订阅者的频道
pubsub numsub [channel] 列出指定频道的订阅者数量
pubsub numpat 列出被订阅者的数量
缓存的使用和设计
缓存的收益和成本
收益:
- 加速读写
- 降低后端负载
成本:
- 数据不一致:缓存层和数据层由时间窗口不一致,和更新策略有关
- 代码维护成本:多了一层缓存逻辑
- 运维成本
使用场景:
- 降低后端负载
- 对高消耗的SQL:join结果集、分组统计结果缓存
- 加速请求响应
- 利用Redis优化IO响应时间
- 大量写合并为批量写
- 计数器先Redis累加再批量写DB
缓存的更新策略
- LRU/LFU/FIFO算法剔除,例如
maxmemory=policy
- 超时剔除,例如
expire
- 主动更新:开发控制生命周期
缓存的粒度控制
- 通用性:全量属性更好
- 占用空间:部分属性更好
- 代码维护:表面上全量属性更好
缓存问题
缓存穿透
cache和数据库中都没有的数据,但是用户不断发起请求。流量大的时候,DB可能宕机。
原因:
- 业务代码自身问题
- 恶意攻击,爬虫
解决方法
- 增加Filter,增加参数合法校验
- 使用布隆过滤器,过滤Redis请求
- 缓存空对象
public String getPassThrought(String key) {
String cacheValue = cache.get(key);
if(StringUtils.isBlank(cacheValue)){
String storageValue = storage.get(key);
cache.set(key,storageValue);
//如果存储数据为空,设置一个过期时间
if(StringUtils.isBlank(storageValue)){
cache.expire(key,60 * 5);
}
return storageValue;
}else {
return cacheValue;
}
}
缓存击穿
cache中没有数据,但是数据库中有数据,某个热点Key突然失效,导致大量请求都打在DB中。
解决方法
- 根据实际场景,考虑热点数据是否可以设置为永不过期。
- 接口限流、服务熔断、降级来保护服务。
缓存雪崩
某时刻大量Key同时失效,流量直接打到DB,导致DB流量瞬间加大,造成级联故障。
解决方法
- 保证缓存高可用性
- 个别节点,个别机器
- Redis Cluster,Redis sentinel
- key的失效时间设置为随机值,避免使用相同的过期时间。
- 依赖隔离组件为后端限流,考虑增加熔断机制。
- 压力测试提前发现问题。
缓存无底洞问题
2010年,FaceBook有3000个Memcache节点,发现问题:加机器没有性能提升,反而下降
问题关键点:
- 更多的节点 != 更高的性能
- 批量接口需求(mget,mset)
- 数据增长和水平扩展需求
优化IO的几种方法
- 命令本身优化,例如慢查询
keys
,hgetall bigkey
- 减少网络通信次数
- 减少接入成本:例如客户端长连接/连接池,NIO等
四种批量优化的方法
- 串行mget
- 串行IO
- 并行IO
- hash_tag
Redis安全
- 设置密码
- 服务端配置:requirepass和masterauth
- 客服端连接:auth命令和-a参数
- 伪装危险命令
- 服务端配置:rename-command为空或者随机字符
- 客户端连接:不可用或者使用指定随机字符
- bind:服务端配置:bind限制的是网卡,并不是客户端IP
- bind不支持config set
- bind 127.0.0.1要谨慎
- 防火墙
- 定期备份
- 不使用默认端口
- 使用非root用户启动