Redis
1 Redis基础
1.1 Redis 简介
- “Redis is an open source (BSD licensed), in-memory data structure store, used as a database, cache and message broker.” —— Redis是一个开放源代码(BSD许可)的内存中数据结构存储,用作数据库,缓存和消息代理。
- Redis 是一个开源,高级的键值存储和一个适用的解决方案,用于构建高性能,可扩展的 Web 应用程序。Redis 也被作者戏称为 数据结构服务器 ,这意味着使用者可以通过一些命令,基于带有 TCP 套接字的简单 服务器-客户端 协议来访问一组 可变数据结构 。(在 Redis 中都采用键值对的方式,只不过对应的数据结构不一样罢了)
1.1.1 Redis 的优点
- 异常快:Redis 非常快,每秒可执行大约 110000 次的设置(SET)操作,每秒大约可执行 81000 次的读取/获取(GET)操作。
- 支持丰富的数据类型 :Redis 支持开发人员常用的大多数数据类型,例如列表,集合,排序集和散列等等。这使得 Redis 很容易被用来解决各种问题,因为我们知道哪些问题可以更好使用地哪些数据类型来处理解决。
- 操作具有原子性 - 所有 Redis 操作都是原子操作,这确保如果两个客户端并发访问,Redis 服务器能接收更新的值。
- 多实用工具 : Redis 是一个多实用工具,可用于多种用例,如:缓存,消息队列(Redis 本地支持发布/订阅),应用程序中的任何短期数据,例如,web应用程序中的会话,网页命中计数等。
1.1.2 Redis 的安装
1.2 Redis 五种基本数据结构
- Redis 有 5 种基础数据结构,它们分别是:string(字符串)、list(列表)、hash(字典)、set(集合) 和zset(有序集合)。这 5 种是 Redis 相关知识中最基础、最重要的部分。
1.2.1 字符串string
SET key value //设置指定 key 的值。
GET key //获取指定 key 的值。
MSET key value [key value ...]//同时设置一个或多个 key-value 对。
MGET key1 [key2..]//获取所有(一个或多个)给定 key 的值。
SETEX key seconds value//将值 value 关联到 key ,并将 key 的过期时间设为 seconds (以秒为单位)。
SETNX key value//只有在 key 不存在时设置 key 的值。
MSETNX key value [key value ...]//同时设置一个或多个 key-value 对,当且仅当所有给定 key 都不存在。
STRLEN key//返回 key 所储存的字符串值的长度。
INCR key //将 key 中储存的数字值增一。
INCRBY key increment //将 key 所储存的值加上给定的增量值(increment)。
DECR key//将 key 中储存的数字值减一。
DECRBY key decrement//key 所储存的值减去给定的减量值(decrement)。
GETSET key value //将给定key的值设为value ,并返回key的旧值(old value)。
/*不常用*/
//GETRANGE key start end 返回 key 中字符串值的子字符
//GETBIT key offset 对key 所储存的字符串值,获取指定偏移量上的位(bit)。
//SETBIT key offset value 对 key 所储存的字符串值,设置或清除指定偏移量上的位(bit)。
//SETRANGE key offset value//用 value 参数覆写给定 key 所储存的字符串值,从偏移量 offset 开始。
//PSETEX key milliseconds value 这个命令和 SETEX 命令相似,但它以毫秒为单位设置 key 的生存时间,而不是像 SETEX 命令那样,以秒为单位。
//INCRBYFLOAT key increment//将key所储存的值加上给定的浮点增量值(increment)。
1.2.2 Hash类型
HDEL key field1 [field2] //删除一个或多个哈希表字段
HEXISTS key field//查看哈希表 key 中,指定的字段是否存在。
HGET key field//获取存储在哈希表中指定字段的值。
HGETALL key//获取在哈希表中指定 key 的所有字段和值
HINCRBY key field increment
//为哈希表 key 中的指定字段的整数值加上增量 increment 。
HKEYS key //获取所有哈希表中的字段
HLEN key //获取哈希表中字段的数量
HMGET key field1 [field2] //获取所有给定字段的值
HMSET key field1 value1 [field2 value2 ]
//同时将多个 field-value (域-值)对设置到哈希表 key 中。
HSET key field value//将哈希表 key 中的字段 field 的值设为 value 。
HSETNX key field value//只有在字段 field 不存在时,设置哈希表字段的值。
HVALS key //获取哈希表中所有值。
//HSCAN key cursor [MATCH pattern] [COUNT count] 迭代哈希表中的键值对。
//HINCRBYFLOAT key field increment为哈希表 key 中的指定字段的浮点数值加上增量 increment 。
1.2.3 List列表
lpush key value1 [value2] //将一个或多个值插入到列表头部(向左压栈)
LRANGE key start stop //获取列表指定范围内的元素
LLEN key//获取列表长度
1.2.4 Set集合
SADD key member1 [member2]//向集合添加一个或多个成员
SCARD key//获取集合的成员数
/*
key1=1,2,3,4
key2= 2,3,6,8
key1=1,2,3,4
*/
SDIFF key1 [key2][key3] //返回第一个集合与其他集合之间的差异。
sdiff key1 key2 //1 4 6 8 => key3(1 4 重复,结果只剩6 8) 6 8
SMEMBERS key//显示key集合下所有元素
sunion key1 [key2]//key1集合和key2集合的并集
sunionstore key3 key1 [key2]//key1集合和key2集合的并集存储到集合key3中
SREM key member1 [member2]//移除集合中一个或多个成员
SDIFFSTORE destination key1 [key2]//返回给定所有集合的差集并存储在 destination 中
1.2.5 Sorted set有序集合
ZADD key score1 member1 [score2 member2]//向有序集合添加一个或多个成员,或者更新已存在成员的分数
redis> ZADD myzset 1 "one"
(integer) 1
redis> ZADD myzset 1 "uno"
(integer) 1
redis> ZADD myzset 2 "two" 3 "three"
(integer) 2
redis> ZRANGE myzset 0 -1 WITHSCORES
1) "one"
2) "1"
3) "uno"
4) "1"
5) "two"
6) "2"
7) "three"
8) "3"
ZCARD key //获取有序集合的成员数
ZINCRBY key increment member//有序集合中对指定成员的分数加上增量 increment
ZREM key member [member ...]//移除有序集合中的一个或多个成员
ZSCORE key member//返回有序集中,成员的分数值
ZREVRANGE key start stop [WITHSCORES]//返回有序集中指定区间内的成员,通过索引,分数从高到低
ZREVRANGEBYSCORE key max min [WITHSCORES]//返回有序集中指定分数区间内的成员,分数从高到低排序
2 Redis常遇到的问题
2.1 缓存穿透
缓存穿透是数据即不再数据库也不再缓存里
缓存穿透的发生一般有两种情况:
- 业务误操作,缓存里和数据库都误删了
- 黑客恶意攻击,故意大量访问某些读取不到的数据的业务
- 解决缓存穿透的方案
- 第一种 增加非空验证
- 第二种设置黑(白)名单
2.2 缓存击穿
指大量请求访问缓存中的一个key时,该key过期了,导致这些请求都去直接访问数据库,短时间大量的请求可能会将数据库击垮
2.3 缓存雪崩
一 般是由大量的key集中在一段时间内失效,或者redis服务故障所引起的,导致查询请求查不到缓存,大量的请求涌入mysql,它会导致mysql瞬间被大量的流量打死,或者是由于该服务接口访问时间过长,导致耗尽了线程资源,不仅会导致该服务其他接口得不到线程资源,也有可能会使上游服务的资源也全都耗在这,然后引起整个系统的崩溃
- 解决方法
- 对于服务不可访问来说,想要一个高可用的服务,首先我们就要集群使用,即使有redis挂了,依然有其他的redis可以过来提供服务,当然最好就是使用redis cluster集群,多个redis master,不仅有主备服务可以在故障时进行切换,即使真的主备都挂了,那也只是一部分数据不能提供服务了,还有其他的master可以给其他提供服务,尽量的减少redis宕机所带来的影响
- 对于大量key同时失效来说最主要的,就是分散key的失效时间,比如说在插入redis缓存的时候,指定key的失效时间为在一段范围区间内的随机值