五大数据类型
官网引用
Redis 是一个开源(BSD许可)的,内存中的数据结构存储系统,它可以用作数据库、缓存和消息中间件。 它支持多种类型的数据结构,如 字符串(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
127.0.0.1:6379> EXPIRE name 10 #设置key的过期时间,单位是秒
(integer) 1
127.0.0.1:6379> ttl name #查看当前key的剩余时间
127.0.0.1:6379> TYPE name #查看key的类型
string
127.0.0.1:6379> MOVE name 1 #将key移动到指定数据库
(integer) 1
127.0.0.1:6379> SELECT 1
OK
127.0.0.1:6379[1]> get name
"lht"
127.0.0.1:6379[1]> SELECT 0
OK
127.0.0.1:6379> get name
(nil)
127.0.0.1:6379[1]> DEL name #删除key
(integer) 1
127.0.0.1:6379[1]> get name
(nil)
String(字符串)
127.0.0.1:6379> set key1 v1 #设置key
OK
127.0.0.1:6379> get key1 #获取key
"v1"
127.0.0.1:6379> APPEND key1 hello #添加内容
(integer) 7
127.0.0.1:6379> get key1
"v1hello"
127.0.0.1:6379> EXISTS key1 #判断是否存在key
(integer) 1
127.0.0.1:6379> strlen key1
(integer) 7
127.0.0.1:6379> APPEND key1 world
(integer) 12
127.0.0.1:6379> STRLEN key1 #获取key的长度
(integer) 12
127.0.0.1:6379> APPEND name zhangsan #如果key不存在,就相当于set
(integer) 8
127.0.0.1:6379> keys *
1) "name"
2) "key1"
##########################################
127.0.0.1:6379> set views 0 #设置初始浏览量为0
OK
127.0.0.1:6379> INCR views #自增1,浏览量加1
(integer) 1
127.0.0.1:6379> get views
"1"
127.0.0.1:6379> INCR views #自减1,浏览量减1
(integer) 2
127.0.0.1:6379> DECR views
(integer) 1
127.0.0.1:6379> INCRBY views 10 #设置步长自增
(integer) 11
127.0.0.1:6379> INCRBY views 10
(integer) 21
127.0.0.1:6379> DECRBY views 10 #设置步长自减
(integer) 11
127.0.0.1:6379> DECRBY views 10
(integer) 1
127.0.0.1:6379> DECRBY views 10
(integer) -9
########################################
127.0.0.1:6379> set name hello,world,redis
OK
127.0.0.1:6379> get name
"hello,world,redis"
127.0.0.1:6379> GETRANGE name 0 3 #截取字符串[0,3]
"hell"
127.0.0.1:6379> GETRANGE name 0 -1 #获取全部的字符串,和get一样
"hello,world,redis"
#########################################
127.0.0.1:6379> set key2 abcdefg
OK
127.0.0.1:6379> SETRANGE key2 1 xx
(integer) 7
127.0.0.1:6379> get key2
"axxdefg"
###########################################
#setex (set with expire) # 设置过期时间
#setnx (set if not exist) # 不存在再设置(再分布式锁中会常常使用!)
127.0.0.1:6379> SETEX key3 30 "hello" #设置key330秒后过期
OK
127.0.0.1:6379> ttl key3
(integer) 26
127.0.0.1:6379> setnx mykey "redis" #如果mykey不存在,创建mykey
(integer) 1
127.0.0.1:6379> ttl key3
(integer) -2
127.0.0.1:6379> SETNX mykey "java" #如果mykey存在,创建失败
(integer) 0
127.0.0.1:6379> get mykey
"redis"
#########################################
127.0.0.1:6379> MSET k1 v1 k2 v2 k3 v3 #同时设置多个值
OK
127.0.0.1:6379> MGET k1 k2 k3 #同时获取多个值
1) "v1"
2) "v2"
3) "v3"
127.0.0.1:6379> MSETNX k1 v1 k4 v4 #msetnx是一个原子性的操作,要么一起成功,要么一起失败
(integer) 0
###########################################
#对象 设置一个user:1独享,值为json字符来保存一个对象
127.0.0.1:6379> MSET user:1:name shangsan user:1:age 22
OK
127.0.0.1:6379> MGET user:1:name user:1:age
1) "shangsan"
2) "22"
##########################################
#先get再set
127.0.0.1:6379> GETSET db redis #如果不存在值,则返回nil
(nil)
127.0.0.1:6379> get db
"redis"
127.0.0.1:6379> GETSET db mysql #如果存在值,获取原来的值,并设置新的值
"redis"
127.0.0.1:6379> get db
"mysql"
String类似的使用场景:value除了是字符串,还可以是数字
- 计数器
- 统计多单位的数量
- 粉丝数
- 对象存储缓存
List
基本的数据类型,列表
在redis里面,list可以实现栈,队列,阻塞队列
#####################################################################
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 3 # 通过区间获取具体的值
1) "three"
2) "two"
3) "one"
127.0.0.1:6379> LRANGE list 0 1
1) "three"
2) "two"
127.0.0.1:6379> RPUSH list four # 将一个或多个值,插入到列表的尾部(右)
(integer) 4
127.0.0.1:6379> LRANGE list 0 -1 # 获取list中的值
1) "three"
2) "two"
3) "one"
4) "four"
#####################################################################
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 0 #通过下标获取list中的某一个值
"two"
127.0.0.1:6379> LINDEX list 1
"one"
#####################################################################
127.0.0.1:6379> LRANGE list 0 -1
1) "two"
2) "one"
127.0.0.1:6379> LLEN list # 返回列表的长度
(integer) 2
#####################################################################
移除指定的值
127.0.0.1:6379> LPUSH list three
(integer) 3
127.0.0.1:6379> LPUSH list three
(integer) 4
127.0.0.1:6379> LRANGE list 0 -1
1) "three"
2) "three"
3) "two"
4) "one"
127.0.0.1:6379> LREM list 1 one #移除指定个数的值
(integer) 1
127.0.0.1:6379> LRANGE list 0 -1
1) "three"
2) "three"
3) "two"
127.0.0.1:6379> LREM list 2 three
(integer) 2
127.0.0.1:6379> LRANGE list 0 -1
1) "two"
#####################################################################
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 # 通过下标截取指定的长度
OK
127.0.0.1:6379> LRANGE mylist 0 -1 # 这个list已经被改变了,截断了只剩下截取的元素
1) "hello1"
2) "hello2"
#####################################################################
移除列表的最后一个元素,将它移动到新的列表中
127.0.0.1:6379> RPOPLPUSH mylist otherlist
"hello3"
127.0.0.1:6379> LRANGE mylist 0 -1
1) "hello"
2) "hello2"
127.0.0.1:6379> LRANGE otherlist 0 -1
1) "hello3"
#####################################################################
lset #将列表中指定下标的值替换为另外一个值
127.0.0.1:6379> LSET list 0 item #将列表中指定下标的值替换为另外一个值,如果不存在列表会报错
(error) ERR no such key
127.0.0.1:6379> LPUSH list v1
(integer) 1
127.0.0.1:6379> LRANGE list 0 -1
1) "v1"
127.0.0.1:6379> LSET list 0 item #如果存在会更新该下标的值
OK
127.0.0.1:6379> LRANGE list 0 -1
1) "item"
127.0.0.1:6379> LSET list 1 other #如果下标不存在,会报错
(error) ERR index out of range
#####################################################################
linsert #将某个具体的value插入到列表中指定元素的前面或后面
127.0.0.1:6379> RPUSH list hello
(integer) 1
127.0.0.1:6379> RPUSH list world
(integer) 2
127.0.0.1:6379> LINSERT list before world other
(integer) 3
127.0.0.1:6379> LRANGE list 0 -1
1) "hello"
2) "other"
3) "world"
127.0.0.1:6379> LINSERT list after world java
(integer) 4
127.0.0.1:6379> LRANGE list 0 -1
1) "hello"
2) "other"
3) "world"
4) "java"
小结
- 实际上是一个链表,before,after,left,right都可以插入元素
- 如果key不存在,创建新的链表
- 如果key存在,新增内容
- 如果移除了所有值,空链表,也代表不存在
- 在两边插入或者改动值下,效率最高,中间元素,相对效率会低一点
消息队列(lpush,rpop),栈(lpush,rpush)
Set(集合)
set中的值是不能重复的
#####################################################################
127.0.0.1:6379> SADD set hello #set集合中添加元素
(integer) 1
127.0.0.1:6379> SADD set world
(integer) 1
127.0.0.1:6379> SADD set zhangsan
(integer) 1
127.0.0.1:6379> SMEMBERS set #查看指定set的所有值
1) "zhangsan"
2) "world"
3) "hello"
127.0.0.1:6379> sismember set hello #判断set是否包含某个元素
(integer) 1
127.0.0.1:6379> sismember set java
(integer) 0
#####################################################################
127.0.0.1:6379> scard set # 返回集合中的元素个数
(integer) 3
#####################################################################
127.0.0.1:6379> srem set hello # 移除set集合中的指定元素
(integer) 1
127.0.0.1:6379> scard set
(integer) 2
127.0.0.1:6379> smembers set
1) "zhangsan"
2) "world"
#####################################################################
set是无序不重复集合
mrandmember
127.0.0.1:6379> srandmember set 1 #随机获取一个元素
1) "world"
127.0.0.1:6379> srandmember set 1
1) "zhangsan"
127.0.0.1:6379> srandmember set 1
1) "mysql"
127.0.0.1:6379> srandmember set 1
1) "mysql"
127.0.0.1:6379> srandmember set 2 #随机获取2个元素
1) "world"
2) "zhangsan"
127.0.0.1:6379> srandmember set 2
1) "world"
2) "zhangsan"
127.0.0.1:6379> srandmember set 2
1) "world"
2) "mysql"
#####################################################################
随机删除key
127.0.0.1:6379> smembers set
1) "mysql"
2) "zhangsan"
3) "world"
127.0.0.1:6379> spop set
"world"
127.0.0.1:6379> spop set
"mysql"
#####################################################################
将一个指定的值,移动到零一个集合中
127.0.0.1:6379> sadd set1 hello
(integer) 1
127.0.0.1:6379> sadd set1 world
(integer) 1
127.0.0.1:6379> sadd set1 java
(integer) 1
127.0.0.1:6379> sadd set2 mysql
(integer) 1
127.0.0.1:6379> SMOVE set1 set2 java
(integer) 1
127.0.0.1:6379> smembers set1
1) "world"
2) "hello"
127.0.0.1:6379> smembers set2
1) "mysql"
2) "java"
#####################################################################
交,并,差集
127.0.0.1:6379> sadd set1 a
(integer) 1
127.0.0.1:6379> sadd set1 b
(integer) 1
127.0.0.1:6379> sadd set1 c
(integer) 1
127.0.0.1:6379> sadd set2 c
(integer) 1
127.0.0.1:6379> sadd set2 d
(integer) 1
127.0.0.1:6379> sadd set2 e
(integer) 1
127.0.0.1:6379> sdiff set1 set2 #差集
1) "b"
2) "a"
127.0.0.1:6379> sinter set1 set2 #交集
1) "c"
127.0.0.1:6379> sunion set1 set2 #并集
1) "e"
2) "c"
3) "a"
4) "b"
5) "d"
微博,A用户将所有关注的人放在一个set集合!将它的粉丝也放在一个集合中
共同关注,共同爱好,二度好友,推荐好友(六度分割理论)
Hash(哈希)
Map集合,key-map,这个时候的值就是一个map集合!本质上和String类型没有太大区别,还是一个简单的key-value!
#####################################################################
127.0.0.1:6379> HSET hash field1 zhangsan #set一个具体的key-value
(integer) 1
127.0.0.1:6379> HGET hash field1 # 获取一个字段值
"zhangsan"
127.0.0.1:6379> HMSET hash field1 hello field2 world #set多个key-value
OK
127.0.0.1:6379> hmget hash field1 field2 #获取多个字段值
1) "hello"
2) "world"
127.0.0.1:6379> hgetall hash #获取全部的数据
1) "field1"
2) "hello"
3) "field2"
4) "world"
#####################################################################
127.0.0.1:6379> hdel hash field1 #删除hash中指定key字段,对应的value值也被删除
(integer) 1
127.0.0.1:6379> hgetall hash
1) "field2"
2) "world"
#####################################################################
127.0.0.1:6379> hlen hash #获取hash的字段数量
(integer) 2
#####################################################################
127.0.0.1:6379> hexists hash field1 # 判断hash中的指定字段是否存在
(integer) 1
127.0.0.1:6379> hexists hash field
(integer) 0
#####################################################################
# 只获取所有的key字段
# 只获取所有的值
127.0.0.1:6379> hkeys hash
1) "field2"
2) "field1"
127.0.0.1:6379> hvals hash
1) "world"
2) "hello"
#####################################################################
incr decr
127.0.0.1:6379> hset hash field3 0
(integer) 1
127.0.0.1:6379> HINCRBY hash field3 1
(integer) 1
127.0.0.1:6379> HINCRBY hash field3 -1
(integer) 0
127.0.0.1:6379> hsetnx hash field4 hello #如果不存在,可以设置
(integer) 1
127.0.0.1:6379> hsetnx hash field4 java #如果存在,不可以设置
(integer) 0
hash变更的数据user,name,age,尤其是用户信息之类的,经常变动的信息,hash更适合对象的存储,String更适合字符串存储
Zset(有序集合)
在set的基础上增加的一个值
#####################################################################
127.0.0.1:6379> zadd myset 1 one
(integer) 1
127.0.0.1:6379> zadd myset 2 two
(integer) 1
127.0.0.1:6379> zadd myset 3 three
(integer) 1
127.0.0.1:6379> ZRANGE myset 0 -1
1) "one"
2) "two"
3) "three
#####################################################################
127.0.0.1:6379> zadd myset 1000 zhangsan
(integer) 1
127.0.0.1:6379> zadd myset 200 lisi
(integer) 1
127.0.0.1:6379> zadd myset 3000 wangwu
(integer) 1
127.0.0.1:6379> ZRANGE myset 0 -1
1) "lisi"
2) "zhangsan"
3) "wangwu"
127.0.0.1:6379> ZRANGEBYSCORE myset -inf +inf withscores #返回全部数据,按升序排列
1) "lisi"
2) "200"
3) "zhangsan"
4) "1000"
5) "wangwu"
6) "3000"
127.0.0.1:6379> ZREVRANGEBYSCORE myset +inf -inf withscores #返回所有数据,按降序排序
1) "wangwu"
2) "3000"
3) "zhangsan"
4) "1000"
5) "lisi"
6) "200"
#####################################################################
zrem,删除一个元素
127.0.0.1:6379> ZRANGE myset 0 -1
1) "lisi"
2) "zhangsan"
3) "wangwu"
127.0.0.1:6379> zrem myset lisi
(integer) 1
127.0.0.1:6379> ZRANGE myset 0 -1
1) "zhangsan"
2) "wangwu"
#####################################################################
127.0.0.1:6379> zcard myset #获取有序集合中的个数
(integer) 2
#####################################################################
127.0.0.1:6379> zadd myset 2000 lisi
(integer) 1
127.0.0.1:6379> zcount myset 1000 2000 #获取指定区间的元素个数
(integer) 2
案例:set排序,存储班级成绩,薪资表排序
带权重进行判断,普通消息:1,重要消息:2
排行榜应用实现