1 Redis数据结构介绍
Redis是一个key-value的数据库,key一般是String类型,不过value的类型多种多样:
基本类型 | 特殊类型 |
---|---|
String | GEO(地理坐标) |
Hash | BitMap |
List | HyperLog |
Set(无重复无序集合) | |
SortedSet(无重复有序集合) |
Redis为了方便学习,将操作不同数据类型的命令也做了分组,在官网(https://redis.io/commands)可以查看到不同的命令:
也可通过命令行查看:
2 Redis通用命令
2.1 KEYS
查看符合模板的所有key,采用模糊查询,不建议在生成环境设备上使用。
语法规则如下:
help [command]
h?llo matches hello, hallo and hxllo
h*llo matches hllo and heeeello
h[ae]llo matches hello and hallo, but not hillo
h[^e]llo matches hallo, hbllo, ... but not hello
h[a-b]llo matches hallo and hbllo
2.2 DEL
删除一个指定的key,返回值为删除的个数。
语法规则如下:
# 可单个/批量删除
DEL key1 [key2 ……]
2.3 EXISTS
判断key是否存在,存在返回1,不存在返回0。
语法规则如下:
EXISTS key
2.4 EXPIRE
给一个key设置有效期,有效期到期时该key会被自动删除。(因为Redis是存储于内存的,若不进行删除清理,则会早日内存爆满)
语法规则如下:
EXPIER key seconds
2.5 TTL
查看一个KEY的剩余有效期。
语法规则如下:
TTL key
返回值有如下三种:
- 正数:剩余秒数
- -1:永久有效
- -2:已经失效
3 String 类型
根据字符串的格式不同,可分为3类:
- string:普通字符串
- int:整数类型,可自增、自减
- float:浮点类型,可自增、自减
不管是哪张格式,地产都是字节数组形式存储,只不过编码方式不同。字符串类型的最大空间不能超过512m。
3.1 String常见命令
String常见命令 | 说明 | 语法格式 |
---|---|---|
set | 添加或修改已经存在的一个String类型的键值对 | set key value |
get | 根据key获取String类型的value | get key |
mset | 批量添加多个String类型的键值对 | mset key value [key value ……] |
mget | 根据多个key获取多个String类型的value | mget key1 [key2 ……] |
incr | 让一个整数的key自增1 | incr key |
incrby | 让一个整数的key自增并指定步长 | incrby key increment |
incrbyfloat | 让一个浮点类型的数字自增并指定步长 | incrbyfloat key increment |
setnx | 添加一个String类型的键值对,前提是这个key不存在,否则不执行(执行成功返回1,否则返回0),效果等同于 set key value nx | setnx key value |
setex | 添加一个String类型的键值对,并指定有效期,效果等同于 set key value ex seconds | setex key seconds value |
4 Redis的key的格式
Redis的key允许有多个单词形成层级结构,多个单词之间用 ’ :’ 隔开,格式并非固定,可根据自己需求删除和添加词条,类似如下:
项目名:业务名:类型:id
如下图所示,创建key:
在桌面客户端进行查看:
5 Hash类型
Hash类型,也叫散列。其value是一个无序字典,类似于Java中的HashMap结构。
我们知道,String结构是将对象序列化为JSON字符串后存储,当需要修改对象的某个字段时是非常不方便的。
Hash结构可以将对象中的每个字段独立存储,可以针对单个字段做CRUD(增查更删)。Hash结构可将JSON中的key和value进行分离管理,如下图所示:
5.1 Hash类型常见命令
Hash常见命令 | 说明 | 语法格式 |
---|---|---|
hset | 添加或修改hash类型key的field的值 | hset key field value [field value ……] |
hget | 获取一个hash类型key的field的值 | hget key field |
hmset | 批量添加多个hash类型key的field的值 | hmset key field value [field value ……] |
hmget | 批量获取多个hash类型key的field的值 | hmget key field [field ……] |
hgetall | 获取一个hash类型的key中的所有的field和value | hgetall key |
hkeys | 获取一个hash类型的key中的所有的field | hkeys key |
hvals | 获取一个hash类型的key中的所有的value | hvals key |
hincrby | 让一个hash类型key的字段值自增并指定步长 | hincrby key field increment |
hsetnx | 添加一个hash类型的key的field值,前提是这个field不存在,否则不执行 | hsetnx key field value |
示例:
6 List类型
Redis中的List类型和Java中的LinkList类似,可以看作是一个双向链表结构(或者是双向队列)。既可支持正向检索,也可支持反向检索。有如下几个特征:
- 有序(和插入顺序有关)
- 元素可重复
- 插入和删除快
- 查询速度一般
常用来存储一个有序数据,例如:朋友圈点赞列表、评论列表等。
6.1 List常用命令
List常用命令 | 说明 | 语法格式 |
---|---|---|
lpush | 向列表左侧插入一个或多个元素 | lpush key element [element ……] |
lpop | 移除并返回列表左侧的第一个元素,没有则返回nil | lpop key [count] |
rpush | 向列表右侧插入一个或多个元素 | rpush key element [element ……] |
rpop | 移除并返回列表右侧的第一个元素,没有则返回nil | rpop key [count] |
lrange | 返回一段角标范围内的所有元素(从0开始) | lrange key start end |
blpop\brpop | 与lpop和rpop类似,只不过在没有元素时等待指定时间,而不是直接返回nil | blpop key [key ……] timeout |
示例:
6.2 思考
(1)如何利用List结构模拟一个栈?
- 入口出口在同一边
- lpush\lpop 或 rpush\rpop
(2)如何利用List结构模拟一个队列?
- 入口出库不同边
- rpush\lpop 或 lpush\rpop
(3)如何利用List结构模拟一个阻塞队列?
- 入口出口不同边
- 出队时采用blpop或brpop
7 Set类型
Redis的Set结构与Java中的HashSet类似,可以看做是一个value为null的HashMap。因此也是一个hash表,因此具备与HashSet类似的特征:
- 无序
- 元素不可重复
- 查找快
- 支持交集、并集、差集等功能
7.1 Set常见命令
set常见命令 | 说明 | 语法格式 |
---|---|---|
sadd | 向set中添加一个或多个元素 | sadd key member [member ……] |
srem | 移除set中的指定元素 | srem key member [member ……] |
scard | 返回set中元素的个数 | scard key |
sismember | 判断一个元素是否存在于set中 | sismember key member |
smembers | 获取set中的所有元素 | smembers key |
sinter key1 key2 …… | 求key1与key2的交集 | |
sdiff key1 key2…… | 求key1与key2的差集 | |
sunion key1 key2 …… | 求key1与key2的并集 |
示例:
8 SortedSet类型
Redis的SortedSet是一个可排序的set集合,与java中的TreeSet有些类似,但底层数据结构却差别很大。SortedSet中每一个元素都带有一个score属性,可用基于score属性对元素排序,底层的实现是一个跳表(SkipList)加 hash 表。
SortedSet具备以下特性:
- 可排序
- 元素不重复
- 查询速度快
因为SortedSet的可排序特性,经常被用来实现排行榜这样的功能。
8.1 SortedSet常见命令
SortedSet常见命令 | 说明 | 语法格式 |
---|---|---|
zadd | 添加一个或多个元素,如果已经存在则更新其score值 | zadd key score member [score member ……] |
zrem | 删除一个指定元素 | zrem key member |
zscore | 获取指定元素的score值 | zscore key member |
zrank | 获取指定元素的排名(从0开始) | zrank key member |
zcard | 获取元素个数 | zcard key |
zcount | 统计score值在给定范围内的所有元素的个数 | zcount key min max |
zincrby | 指定元素自增,自定义步长 | zincrby key increment member |
zrange | 按照score排序后,获取指定排名范围内的元素 | zrange key min max |
zrangebyscore | 按照score排序后,获取指定score范围内的元素 | zrangebyscore key min max |
zdiff、zinter、zunion | 求差集、交集、并集 |
注意:所有的排名默认都是升序,如果要降序则在命令的Z后面+REV即可。
8.2 SortedSet命令练习
将班级的下列学生得分存入Redis的SortedSet中:
Jack 85,Lucy 89,Rose 82,Tom 95,Jerry 78,Amy 92,Miles 76
并实现下列功能:
首先,建立班级学生的信息:
(1)删除Tom同学
(2)获取Amy同学的分数
(3)获取Rose同学的排名
(4)查询80分以下有几个同学
(5)给Amy同学加2分
(6)查出成绩前3名的同学
(7)查出成绩80分以下的所有同学