一、Redis由来
redis之父:antirez,意大利人,开发于2008年
产生原因:当时的antirez开发一个统计所有访问情况(ip、操作系统、浏览器等所有网站的访问信息)的网站
但数据怎么存储呢?
当时作者用的是MySQL数据库,由于MySQL数据是存放在磁盘中,当高并发,数据量大时,无论怎么优化都很慢
antirez想到把数据存放到内存中,可对数据加上持久化的功能,发布于09年 —redis(remote dictionary service 远程字典服务)
redis端口:6379(小故事:手机键盘代表:MERZ,当时的一个广告明星【暗语:愚蠢】)
二、Redis特性
1、数据操作都在内存
2、命令执行的单线程(6.0以后支持多线程【但也只是把网络开销、fork持久化操作放入了多线程】)
redis瓶颈不在内存,而是CPU,如果使用多线程,线程切换反而会降低效率,所有单线程更快
3、数据结构:key-value
4、功能丰富(持久化)
5、IO多路复用
钓鱼为例:
一个鱼竿钓鱼,人等待鱼上钩的过程,可理解阻塞IO
10个鱼竿钓鱼,人对每个鱼竿轮巡查看的过程,可理解为非阻塞IO
鱼上钩鱼竿是有动静的,人观察鱼竿状态的过程,可理解为IO多路复用
注:key最大值512MB,value最大值分情况String最大值512M,list,set,zset最大值-1 elements(元素)
redis默认有16个库(可配置) ,下标0-15,默认使用下标是0的库
每个库之间数据是隔离的(可以包含相同的key值)
分库不适合对业务数据进行隔离,因为redis有相关命令可以操作全库,有可能对数据完整性进行破坏
如果需要对业务数据隔离,就部署不同的服务器,或相同服务器不同端口
redis安装步骤:https://blog.youkuaiyun.com/chai1143126501/article/details/113853986
三、Redis常见的数据类型以及应用场景
1、String常用命令
set [key] [value] | 添加key-value值 |
get [key] | 获取value值 |
mset [key1] [value2] [key1] [value2] …… | 批量添加key-value值 |
mget [key] [key]…… | 批量获取value值 |
strlen [key] | 获取value长度 |
append [key] [value2] | 对key值追加新串 |
getrange [key] [start] [end] | 截取字符串,start、end都是下标索引值(范围) |
rename [key] [newKey] | 修改key值 / 对key重命名 |
incr [key] | 数值自增(对应的value必须是数字),可以做主键(分布式ID)的原因,还有就是redis是单线程执行命令,不用担心并发会造成ID冲突 |
incrby [key] [step] | 数值自增步长,根据step值进行自增 |
incrbyfloat [key] [step] | 数值浮点数自增步长 |
setnx [key] [value] | 分布式锁,第一次设值成功后,第二次再设值失败(值存在再设值就会失败) |
set [key] [value] EX [seconds] NX|XX | 分布式锁,EX:过期时间(单位秒);NX:值不存在才设值成功,XX:值存在设值才成功 |
使用场景:
incr 分布式ID,自增 统计数据(浏览数,点赞数)
setnx 分布式锁
注:在Java使用过程中,分布式锁要加过期时间,但两个命令执行要保证原子性,否则就会有并发问题
使用lua脚本能保证多条命令的原子性,并减少执行命令的网络开销
执行lua脚本案例:
eval "redis.call('set',KEYS[1],ARGV[1]) return KEYS[1]" 1 zhangsan 18
2、hash常用命令
hset [key] [field] [value] | 单个key,设置field对应value,field可理解为小key[字段|属性] |
hmset [key] [field1] [value1] [field2] [value2] …… | 单个key,设置多个field对应value |
hget [key] [field] | 获取单个key的field值 |
hmget [key] [field1] [field2] | 获取单个key的多个field值 |
hgetall [key] | 获取单个key的所有值 |
hincrby [key] [field] [step] | 对key的field对应值+步长(step) |
hexists [key] [field] | 判断field是否存在,返回值:1(存在),返回值:0(不存在) |
hdel [key] [field] | 删除field,返回值:1(删除成功),返回值:0(删除失败) |
hlen [key] | 返回key的size,field个数 |
使用场景:
经典场景:购物车,用户ID作为key,商品ID作为Filed,数量作为value
3、list常用命令
特性:从左到右有序,可重复
lpush [key] [value1] [value2] …… | 从集合左边添加数据 |
lrange [key] [start] [end] | 查看集合中的区间值,下标start至下标end,如果是0 -1是显示全部 |
rpush [key] [value1] [value2] …… | 从集合右边添加数据 |
lpop [key] | 从集合左边输出一条数据(输出后会从队列中删除) |
rpop [key] | 从集合右边输出一条数据(输出后会从队列中删除) |
blpop [key1] [key2] …… [timeout] | 从集合左边输出一条数据(输出后会从队列中删除),如果集合中没有数据则阻塞等待多少秒,超时退出,timeout超时时间(单位秒) |
brpop [key1] [key2] …… [timeout] | 从集合右边输出一条数据(输出后会从队列中删除),如果集合中没有数据则阻塞等待多少秒,超时退出,timeout超时时间(单位秒) |
使用场景:
可以做队列,但不建议做队列,因为不能保证数据及时持久化(不能保证数据不丢失)
因为有序可以做朋友圈,按时间顺序输出数据条目,可之间进行排序
4、set常用命令
特性:不重复,无序
sadd [key] [value1] [value2] …… | 添加集合数据 |
smembers [key] | 查看集合所有数据(显示的数据顺序不是添加的先后顺序,存储无序;但显示数据顺序固定后,获取数据时按顺序输出) |
scard [key] | 查看集合长度 |
srandmember [key] [count] | 随机查看集合中多少条数据;count 数值,多少条 |
spop [key] [count] | 随机输出集合中多少条数据,输出后该条数据会被删除;count 数值,多少条,如果没写数值,默认输出1条 |
sismember [key] [value] | 判断集合中是否存在value值;如果存在,返回值:1;如果不存在,返回值:0 |
sdiff [key1] [key2] …… | 查看后面集合不在前面集合中出现的集合数据;例如:A(1,2,3,4),B(3,4,5),输出集合:1,2 |
sinter [key1] [key2] …… | 查看集合的交集;例如:A(1,2,3,4),B(3,4,5),输出集合:3,4 |
sunion [key1] [key2] …… | 查看集合的并集;例如:A(1,2,3,4),B(3,4,5),输出集合:1,2,3,4,5 |
应用场景:
无序:抽奖
判断是否存在:点赞、签到
交集/并集 :共同好友,可能认识的人
5、zset常用命令
特性:有序集合,对value值有个score(分数)参数,如果score参数相同,比较ASCII码决定先后顺序
zadd [key] [score1] [value1] [score2] [value2] …… | 添加集合数据,score代表值的先后顺序 |
zrange [key] [start] [end] | 查看集合中的区间值,下标start至下标end,如果是0 -1是显示全部,并且集合根据score从小到大排序 |
zrevrange [key] [start] [end] | 查看集合中的区间值,下标start至下标end,如果是0 -1是显示全部,并且集合根据score从大到小排序 |
zrangebyscore [key] [start] [end] | 查看根据score值的区间集合,比如获取score[10,20]之间的集合 |
zrem [key] [value1] [value2]…… | 删除集合中的值 |
zscore [key] [value] | 获取值的score |
zincrby [key] [step] [value] | 对值的score增加步长 |
zcount [key] [start] [end] | 统计集合中score区间有多少条数据 |
zrank [key] [value] | 返回value的索引(下标)位置 |
使用场景:
统计:排名/排行榜
分数,有序:取前多少名
四、redis其他常用命令
常用命令 | 注释 |
select [index] | 选库 / 切库 |
flushall | 全库清空 |
set [key] [value] | 添加String值 |
get [key] | 获取String值 |
mset [key] [value] [key] [value] …… | 批量添加String值 |
mget [key] [key]…… | 批量获取String值 |
rename [key] [newKey] | 修改key值 / 对key重命名 |
keys * | 获取所有库中的key(不要在生产环境下使用,如果redis库中数据量大,会卡死,甚至宕机) |
expire [key] [seconds] | 对key设置过期时间,单位秒 |
ttl [key] | 查看key过期时间,key已过期返回-2,key未设置过期时间返回-1 |
del [key] | 删除key |
redis命名整理参考:http://redisdoc.com/index.html |