redis五大数据类型
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-tx2Oq4gM-1620556205638)(E:\笔记整合\NoSQL\redis官网.png)]
Redis 是一个开源(BSD许可)的,内存中的数据结构存储系统,它可以用作数据库、缓存和消息中间件MQ。 它支持多种类型的数据结构,如 字符串(strings), 散列(hashes), 列表(lists), 集合(sets), 有序集合(sorted sets) 与范围查询, bitmaps, hyperloglogs 和 地理空间(geospatial) 索引半径查询。 Redis 内置了 复制(replication),LUA脚本(Lua scripting), LRU驱动事件(LRU eviction),事务(transactions) 和不同级别的 磁盘持久化(persistence), 并通过 Redis哨兵(Sentinel)和自动 分区(Cluster)提供高可用性(high availability)。
Redis-key
keys * 查看所有的key
set name laowang set key
EXISTS name 判断当前的key是否存在
move name 1 移动key指定的数据库
EXPIRE name 10 设置key的过期时间
ttl name 查看当前key的剩余时间
get name 查看key
type bane 查看当前key的类型
不会可以查看官方文档
String(字符串)
set key1 v1 #设置值
get key1 #获取值
keys * #获取所有的key
exists key1 #查看key是否存在
append key1 "hello" #追加字符,如果不存在就相当于set
strlen key1 #查看字符串长度
expire key1 10 #设置过期时间
ttl key1 #查看剩余时间
=======================================================
set views 0 #初始浏览量为0
incr views #自增1
decr views #自减1
incrby views 10 #可以设置步长,指定增量
decrby views 5 #
=======================================================
set key1 "hello,laowang"
get key1
getrange key1 0 3 #截取字符转 [0,3]
getrange key1 0 -1 #获取全部字符串 和 get key1是一样的
=======================================================
setrange key1 1 xx #替换指定位置开始的字符串
=======================================================
setex(set with expire) #设置过期时间
setnx(set if not exist) #不存在再设置(在分布式锁中会常常使用)
setex key3 30 "hello" #设置key3的值为hello,30s后过期
ttl key3
setnx key4 "redis" #如果key4不存在,就创建
setnx key4 "mongoDB" #设置失败,key4存在,创建失败
======================================================
mset k1 v1 k2 v2 k3 v3 #批量创建k1、k2、k3
mget k1 k2 k3 #批量获取k1、k2、k3
msetnx k1 v1 k4 v4 #是一个原子性操作,如果都不存在,批量创建,如果有某些值存在就失败
#对象
mset user:2 {name:lisi,age:3} #这里设置了一个对象user:2,值为json字符来保存一个对象
OK
mget user:2
"{name:lisi,age:3}"
#这里的key是一个巧妙的设置,user:{id}:{filed},如此设置在redis是允许的
mset user:1:name zhangsan user:1:age 2
OK
mget user:1:name user:1:age
"zhangsan"
"2"
========================================================
getset #先get然后再set
getset name laowang #如果不存在就返回nil,并创建
(nil)
get name
"laowang"
getset name xiaopang #如果存在就返回旧值,并更新为新值
"laowang"
get name
"xiaopang"
String类型的使用场景:value除了是字符串还可以是数字
- 计数器
- 统计多单位的数量
- 粉丝数
- 对象缓存存储
Lists(列表)
可以实现栈、队列、阻塞队列
命令都是以l开头
127.0.0.1:6379> lpush list one #将一个或者多个值插入到列表头部
(integer) 1
127.0.0.1:6379> lpush list two
(integer) 2
127.0.0.1:6379> lpush list three
(integer) 3
127.0.0.1:6379> lrange list 0 -1 #获取list中的值
1) "three"
2) "two"
3) "one"
127.0.0.1:6379> rpush list four #将一个或多个值插入到列表尾部
(integer) 4
127.0.0.1:6379> lrange list 0 -1
1) "three"
2) "two"
3) "one"
4) "four"
=======================================================================
127.0.0.1:6379> lpop list #移除列表的第一个元素
"three"
127.0.0.1:6379> rpop list #移除列表的最后一个元素
"four"
127.0.0.1:6379> lrange list 0 -1
1) "two"
2) "one"
=======================================================================
127.0.0.1:6379> lindex list 1 #通过下标获取list中的值
"one"
127.0.0.1:6379> lindex list 0
"two"
=======================================================================
127.0.0.1:6379> llen list #查询list长度
(integer) 2
127.0.0.1:6379> lrange list 0 -1
1) "two"
2) "one"
127.0.0.1:6379> lpush list two
(integer) 3
127.0.0.1:6379> lrange list 0 -1
1) "two"
2) "two"
3) "one"
127.0.0.1:6379> lrem list 1 two #移除list中指定个数的value,精确匹配
(integer) 1
127.0.0.1:6379> lrange list 0 -1
1) "two"
2) "one"
127.0.0.1:6379> lpush list two
(integer) 3
127.0.0.1:6379> lrem list 2 two
(integer) 2
127.0.0.1:6379> lrange list 0 -1
1) "one"
========================================================================
127.0.0.1:6379> rpush mylist "hello"
(integer) 1
127.0.0.1:6379> rpush mylist "hello1"
(integer) 2
127.0.0.1:6379> rpush mylist "hello2"
(integer) 3
127.0.0.1:6379> rpush mylist "hello3"
(integer) 4
127.0.0.1:6379> ltrim mylist 1 2 #通过下标截取指定的长度,这个list已经被改变了,截取了只剩下截取的元素
OK
127.0.0.1:6379> lrange mylist 0 -1
1) "hello1"
2) "hello2"
=========================================================================
127.0.0.1:6379> rpush mylist "hello"
(integer) 1
127.0.0.1:6379> rpush mylist "hello1"
(integer) 2
127.0.0.1:6379> rpush mylist "hello2"
(integer) 3
127.0.0.1:6379> rpoplpush mylist myotherlist #移除列表的最后一个元素,将他移动到新的列表中
"hello2"
127.0.0.1:6379> lrange mylist 0 -1
1) "hello"
2) "hello1"
127.0.0.1:6379> lrange myotherlist 0 -1
1) "hello2"
===========================================================================
127.0.0.1:6379> exists list #判断列表是否存在
(integer) 0
127.0.0.1:6379> lset list 0 item #如果不存在列表我们去更新就会报错
(error) ERR no such key
127.0.0.1:6379> lpush list value1
(integer) 1
127.0.0.1:6379> lrange list 0 0
1) "value1"
127.0.0.1:6379> lset list 0 item #如果存在,更新当前下表的值
OK
127.0.0.1:6379> lrange list 0 0
1) "item"
127.0.0.1:6379> lset list 1 other #如果不存在就会报错
(error) ERR index out of range
============================================================================
127.0.0.1:6379> rpush mylist world
(integer) 1
127.0.0.1:6379> linsert mylist before world other #将other插入world的前面
(integer) 2
127.0.0.1:6379> lrange mylist 0 -1
1) "other"
2) "world"
127.0.0.1:6379> linsert mylist after other two #将two插入other的后面
(integer) 3
127.0.0.1:6379> lrange mylist 0 -1
1) "other"
2) "two"
3) "world"
总结
-
实际那个list是一个链表,before Node after ,left right 都可以插入值
-
如果key不存在,创建新的链表
-
如果key存在,新增内容
-
如果移除了所有值,空链表,也代表不存在
-
在两边插入或者改动值,效率最好!在中间元素,效率低
消息队列 Lpush Rpop ,栈 Rpush Lpop
Set(集合)
set中的值是不能重复的
127.0.0.1:6379> sadd myset hello #set集合中添加元素
(integer) 1
127.0.0.1:6379> sadd myset laowang
(integer) 1
127.0.0.1:6379> sadd myset lovelaowang
(integer) 1
127.0.0.1:6379> smembers myset #查看指定set中所有值
1) "lovelaowang"
2) "laowang"
3) "hello"
127.0.0.1:6379> sismember myset hello #判断set中是不是包含该值
(integer) 1
127.0.0.1:6379> sismember myset xiaopang
(integer) 0
127.0.0.1:6379> scard myset #查看set集合中的元素个数
(integer) 3
127.0.0.1:6379> srem myset hello #删除指定的元素
(integer) 1
127.0.0.1:6379> smembers myset
1) "lovelaowang"
2) "laowang"
========================================================
127.0.0.1:6379> srandmember myset #随机选取集合中一个元素
"laowang"
127.0.0.1:6379> srandmember myset 2 #随机选取集合中两个元素
1) "lovelaowang"
2) "laowang"
127.0.0.1:6379> clear
127.0.0.1:6379> smembers myset
1) "lovelaowang"
2) "laowang"
127.0.0.1:6379> spop myset #随机删除一个元素
"lovelaowang"
127.0.0.1:6379> smembers myset
1) "laowang"
========================================================
127.0.0.1:6379> sadd myset hello
(integer) 1
127.0.0.1:6379> sadd myset world
(integer) 1
127.0.0.1:6379> sadd myset2 set2
(integer) 1
127.0.0.1:6379> smove myset myset2 world #将指定的元素转移到指定的set集合
(integer) 1
127.0.0.1:6379> smembers myset
1) "hello"
127.0.0.1:6379> smembers myset2
1) "set2"
2) "world"
========================================================
共同关注 交集
差集
127.0.0.1:6379> sadd key1 a
(integer) 1
127.0.0.1:6379> sadd key1 b
(integer) 1
127.0.0.1:6379> sadd key1 c
(integer) 1
127.0.0.1:6379> sadd key2 c
(integer) 1
127.0.0.1:6379> sadd key2 d
(integer) 1
127.0.0.1:6379> sadd key2 e
(integer) 1
127.0.0.1:6379> sdiff key1 key2 #差集
1) "a"
2) "b"
127.0.0.1:6379> sinter key1 key2 #交集
1) "c"
127.0.0.1:6379> sunion key1 key2 #并集
1) "c"
2) "d"
3) "a"
4) "b"
5) "e"
微博,A用户将所有关注的人放在一个set集合中,将它的粉丝也放在一个集合中
共同关注,共同爱好,二度好友,推荐好友!(流毒分割理论)
Hash(哈希)
Map集合,key-map
本质和String类型没有太大的区别,还是一个简单的key-value
127.0.0.1:6379> hset myhash field1 laowang #set一个具体的key-value
(integer) 1
127.0.0.1:6379> hget myhash field1 #获取字段值
"laowang"
127.0.0.1:6379> hmset myhash field2 hello field1 world #set多个具体的key-value
OK
127.0.0.1:6379> hmget myhash field1 field2 #获取多个字段值
1) "world"
2) "hello"
127.0.0.1:6379> hgetall myhash #获取全部数据
1) "field1"
2) "world"
3) "field2"
4) "hello"
127.0.0.1:6379> hdel myhash field1 #删除hash的指定key字段!对应的value也就消失了
(integer) 1
127.0.0.1:6379> hgetall myhash
1) "field2"
2) "hello"
========================================================
127.0.0.1:6379> hgetall myhash
1) "field2"
2) "hello"
3) "field1"
4) "laowang"
127.0.0.1:6379> hexists myhash field1 #判断指定字段是否存在
(integer) 1
127.0.0.1:6379> hexists myhash field3
(integer) 0
========================================================
127.0.0.1:6379> hkeys myhash #只获取所有的field
1) "field2"
2) "field1"
127.0.0.1:6379> hvals myhash #只获取所有的value
1) "hello"
2) "laowang"
========================================================
127.0.0.1:6379> hset myhash field3 5
(integer) 1
127.0.0.1:6379> hincrby myhash field3 1 #指定增量
(integer) 6
127.0.0.1:6379> hincrby myhash field3 -1
(integer) 5
127.0.0.1:6379> hsetnx myhash field4 world #如果不存在就设置
(integer) 1
127.0.0.1:6379> hkeys myhash
1) "field2"
2) "field1"
3) "field3"
4) "field4"
hash变更的数据user name age,尤其是用户信息子之类,经常变动的信息!hash更适合于对象的存储,String更加适合字符串存储
Zset(有序集合)
在set的基础上,添加了一个值,set k1 v1 , zset k1 score1 v1
127.0.0.1:6379> zadd myset 1 one #set值
(integer) 1
127.0.0.1:6379> zadd myset 2 two 3 three
(integer) 2
127.0.0.1:6379> zrange myset 0 -1
1) "one"
2) "two"
3) "three"
127.0.0.1:6379> zadd salary 2500 xiaohong
(integer) 1
127.0.0.1:6379> zadd salary 5000 zhangsan
(integer) 1
127.0.0.1:6379> zadd salary 500 laowang
(integer) 1
127.0.0.1:6379> zrangebyscore salary -inf +inf #从小到大排序
1) "laowang"
2) "xiaohong"
3) "zhangsan"
127.0.0.1:6379> zrangebyscore salary -inf +inf withscores #从小到大排序带上scores
1) "laowang"
2) "500"
3) "xiaohong"
4) "2500"
5) "zhangsan"
6) "5000"
127.0.0.1:6379> zrangebyscore salary -inf 2500 withscores #从小到大 到2500
1) "laowang"
2) "500"
3) "xiaohong"
4) "2500"
127.0.0.1:6379> zrevrangebyscore salary +inf -inf withscores #从大到小
1) "zhangsan"
2) "5000"
3) "xiaohong"
4) "2500"
5) "laowang"
6) "500"
127.0.0.1:6379> zrevrange salary 0 -1 #从大到小排序
1) "zhangsan"
2) "xiaohong"
========================================================
127.0.0.1:6379> zrange salary 0 -1
1) "laowang"
2) "xiaohong"
3) "zhangsan"
127.0.0.1:6379> zrem salary laowang #移除zset中的指定元素
(integer) 1
127.0.0.1:6379> zrange salary 0 -1
1) "xiaohong"
2) "zhangsan"
127.0.0.1:6379> zcard salary #获取集合中的元素
(integer) 2
========================================================
127.0.0.1:6379> zadd myset 1 hello
(integer) 1
127.0.0.1:6379> zadd myset 2 world 3 laowang
(integer) 2
127.0.0.1:6379> zcount myset 1 3 #获取指定范围大小内集合中的个数
(integer) 3
127.0.0.1:6379> zcount myset 1 2
(integer) 2
127.0.0.1:6379> zadd myset 6 xxx
(integer) 1
127.0.0.1:6379> zcount myset 1 4
(integer) 3
127.0.0.1:6379> zcount myset 1 10
(integer) 4
案例思路:set 排序 存储班级成绩信息,工资表排序
普通消息 1 重要消息 2 带权重进行判断
排行榜应用实现,取Top N测试