Redis常用命令

Redis常用命令

官网:https://www.redis.net.cn/

1、客户端连接

$ redis-cli -p 6379

Redis常用命令_第1张图片

执行shutdownredis server会关闭。

2、Redis性能测试

redis-benchmark 是一个压力测试工具!

官方自带的性能测试工具!

Redis 性能测试是通过同时执行多个命令实现的。

redis 性能测试的基本命令如下:

$ redis-benchmark [option] [option value]

注意:该命令是在 redis 的目录下执行的,而不是 redis 客户端的内部指令。

以下实例同时执行 100000 个请求来检测性能:

# 测试:100个并发连接 100000请求
$ redis-benchmark -h localhost -p 6379 -c 100 -n 100000

Redis常用命令_第2张图片

# 输出结果分析
# 对我们的100000个请求进行写入测试
100000 requests completed in 1.30 seconds
# 100个并发客户端
100 parallel clients
# 每次写入3个字节
3 bytes payload
# 只有一台服务器来处理这些请求,单机性能
keep alive: 1

# 所有请求在3毫秒处理完成
36.06% <= 1 milliseconds
97.24% <= 2 milliseconds
99.99% <= 3 milliseconds
100.00% <= 3 milliseconds
# 每秒处理76745.97次请求
76745.97 requests per second

redis 性能测试工具可选参数如下所示:

序号 选项 描述 默认值
1 -h 指定服务器主机名 127.0.0.1
2 -p 指定服务器端口 6379
3 -s 指定服务器 socket
4 -c 指定并发连接数 50
5 -n 指定请求数 10000
6 -d 以字节的形式指定 SET/GET 值的数据大小 2
7 -k 1=keep alive 0=reconnect 1
8 -r SET/GET/INCR 使用随机 key, SADD 使用随机值
9 -P 通过管道传输 请求 1
10 -q 强制退出 redis。仅显示 query/sec 值
11 –csv 以 CSV 格式输出
12 -l(L 的小写字母) 生成循环,永久执行测试
13 -t 仅运行以逗号分隔的测试命令列表。
14 -I(i 的大写字母) Idle 模式。仅打开 N 个 idle 连接并等待。

3、切换数据库

redis默认有16个数据库,默认使用的是第0个。

可以使用 select 进行切换数据库!

127.0.0.1:6379> select 3 # 切换数据库 
OK
127.0.0.1:6379[3]> DBSIZE # 查看DB大小! 
(integer) 0
127.0.0.1:6379[3]> keys * # 查看数据库所有的key 
1) "name"
127.0.0.1:6379[3]> flushdb 
OK
127.0.0.1:6379[3]> keys * 
(empty list or set)

清除当前数据库 flushdb

清除全部数据库的内容 FLUSHALL

4、Redis-Key

127.0.0.1:6379> set name helloworld # set key 
OK
127.0.0.1:6379> keys * 
1) "name"
127.0.0.1:6379> EXISTS name # 判断当前的key是否存在 
(integer) 1 
127.0.0.1:6379> EXISTS name1 
(integer) 0 
127.0.0.1:6379> move name 1 # 移除当前的key 
(integer) 1
127.0.0.1:6379> clear # 清空当前屏幕
127.0.0.1:6379> get name 
"helloworld" 
127.0.0.1:6379> EXPIRE name 10 # 设置key的过期时间,单位是秒 
(integer) 1 
127.0.0.1:6379> ttl name # 查看当前key的剩余时间 
(integer) 4
127.0.0.1:6379> get name 
(nil) 
127.0.0.1:6379> type name # 查看当前key的一个类型! 
string

5、String

set key value 添加键值对 
get <key> 查询对应的键值对
append <key><value> 将给定的<value>追加到原址的末尾
strlen <key> 获取值的长度
setnx <key><value> 只有在key不存在时设置key的值
setex <key><过期时间><value>
	-设置键的同时,设置过期时间,单位秒
incr <key>
	-将key中储存的数字值增1
	-只能对数字值操作,如果为空,新增值为1
decr <key>
	-将key中储存的数字值减1
	-只能对数字操作,如果为空,新增值为-1
incrby/decrby <key><步长> 将Key中储存的数字值增减。自定义步长
mset <key1><value1><key2><value2> 同时设置一个或多个key-value键值对
mget <key1><key2><key3> 同时获取一个或多个value
msetnx <key1><value1><key2><value2>...
	-同时设置一个或多个key-value对,当且仅当所给定的key不存在时
	-原子性,有一个失败则全都失败
getrange <key><起始位置><结束位置>
	-获得值的范围,雷士java中的substring,前包,后包
setrange <key><起始位置><value>
	-用<value>覆写<key>所储存的字符串的值,从<起始位置>开始(索引从0开始)
getset <key><value> 以旧换新,设置了新值的同时获得旧值。

set user:1 {name:zhangsan,age:20} 设置一个user:1对象 值为json字符串来保存一个对象
127.0.0.1:6379> set key1 v1 # 设置值 
OK
127.0.0.1:6379> get key1 # 获得值 
"v1" 
127.0.0.1:6379> keys * # 获得所有的key 
1) "key1" 
127.0.0.1:6379> EXISTS key1 # 判断某一个key是否存在 
(integer) 1 
127.0.0.1:6379> APPEND key1 "hello" # 追加字符串,如果当前key不存在,就相当于setkey 
(integer) 7 
127.0.0.1:6379> get key1 
"v1hello" 
127.0.0.1:6379> STRLEN key1 # 获取字符串的长度! 
(integer) 7
127.0.0.1:6379> GETRANGE key1 0 3 # 截取字符串 [0,3]
127.0.0.1:6379> GETRANGE key1 0 -1 # 获取全部的字符串和get key是一样的
127.0.0.1:6379> SETRANGE key1 1 xx # 替换指定位置开始的字符串
# setex (set with expire) 设置过期时间 
# setnx (set if not exist) 不存在在设置
127.0.0.1:6379> setex key3 30 "hello" # 设置key3 的值为 hello,30秒后过期 
OK
127.0.0.1:6379> ttl key3 
(integer) 26 
127.0.0.1:6379> get key3 
"hello" 
127.0.0.1:6379> setnx mykey "redis" # 如果mykey 不存在,创建mykey
(integer) 1
127.0.0.1:6379> setnx mykey "MongoDB" # 如果mykey存在,创建失败! 
(integer) 0
127.0.0.1:6379> mset k1 v1 k2 v2 k3 v3 # 同时设置多个值 
OK
127.0.0.1:6379> keys * 
1) "k1" 
2) "k2" 
3) "k3" 
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 
127.0.0.1:6379> get k4 
(nil)
# 对象 
set user:1 {name:zhangsan,age:3} # 设置一个user:1 对象,值为json字符来保存一个对象!
# 这里的key是一个巧妙的设计:user:{id}:{filed}, 如此设计在Redis中是完全OK了! 
127.0.0.1:6379> mset user:1:name zhangsan user:1:age 2 
OK
127.0.0.1:6379> mset user:1:name zhangsan user:1:age 2 
OK
127.0.0.1:6379> mget user:1:name user:1:age 
1) "zhangsan" 
2) "2"
# getset 先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 mongodb # 如果存在值,获取原来的值,并设置新的值 
"redis" 
127.0.0.1:6379> get db 
"mongodb"
127.0.0.1:6379> set views 0 # 初始浏览量为0 
OK
127.0.0.1:6379> get views 
"0" 
127.0.0.1:6379> incr views # 自增1 浏览量变为1 
(integer) 1 
127.0.0.1:6379> incr views 
(integer) 2 
127.0.0.1:6379> get views 
"2" 
127.0.0.1:6379> decr views # 自减1 浏览量-1 
(integer) 1 
127.0.0.1:6379> decr views 
(integer) 0 
127.0.0.1:6379> decr views 
(integer) -1 
127.0.0.1:6379> get views 
"-1" 
127.0.0.1:6379> INCRBY views 10 # 可以设置步长,指定增量! 
(integer) 9 
127.0.0.1:6379> INCRBY views 10 
(integer) 19 
127.0.0.1:6379> DECRBY views 5 
(integer) 14

6、List

lpush/rpush <key><value1><value2><value3>...从左边/右边插入一个或多个值
lpop/rpop <key> 从左边/右边吐出一个值。值在键在,值光键消
rpoppush <key1><key2><key1>列表右边吐出来一个值,插到<key2>列表右边
rpoplpush <key1><key2> 移除列表最后一个元素,将其添加到新的列表
lrange <key><start><stop> 按照索引获得元素(从左到右)
lindex <key><index> 按照所给下标获得元素(从左到右)
llne <key> 获取列表的长度 
linsert<key> before | after<value><newvalue><value>的后面插入<newvalue>插入值
lrem<key><n><value>从左边删除n个value(从左到右)
lset<key><index><value>将列表key下标为<index>的值替换成<value>
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> LRANGE list 0 1 # 通过区间获取具体的值! 
1) "three" 
2) "two" 
127.0.0.1:6379> Rpush list righr # 将一个值或者多个值,插入到列表位部 (右) 
(integer) 4 
127.0.0.1:6379> LRANGE list 0 -1 
1) "three" 
2) "two" 
3) "one" 
4) "righr"
# LPOP RPOP
127.0.0.1:6379> LRANGE list 0 -1 
1) "three" 
2) "two" 
3) "one" 
4) "righr" 
127.0.0.1:6379> Lpop list # 移除list的第一个元素 
"three" 
127.0.0.1:6379> Rpop list # 移除list的最后一个元素 
"righr" 
127.0.0.1:6379> LRANGE list 0 -1 
1) "two" 
2) "one"
# Lindex
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"
# Llen
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> Llen list # 返回列表的长度 
(integer) 3
# Lrem
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 # 移除list集合中指定个数的value,精确匹配 
(integer) 1 
127.0.0.1:6379> LRANGE list 0 -1 
1) "three" 
2) "three" 
3) "two" 
127.0.0.1:6379> lrem list 1 three 
(integer) 1 
127.0.0.1:6379> LRANGE list 0 -1 
1) "three" 
2) "two" 
127.0.0.1:6379> Lpush list three 
(integer) 3 
127.0.0.1:6379> lrem list 2 three 
(integer) 2 
127.0.0.1:6379> LRANGE list 0 -1 
1) "two"
# trim
127.0.0.1:6379> keys * 
(empty list or set) 
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"
# rpoplpush 移除列表最后一个元素,将其添加到新的列表
127.0.0.1:6379> rpush list "hello"
(integer) 1
127.0.0.1:6379> rpush list "hello1"
(integer) 2
127.0.0.1:6379> rpush list "hello2"
(integer) 3
127.0.0.1:6379> rpush list "hello3"
(integer) 4
127.0.0.1:6379> rpoplpush list list1
"hello3"
127.0.0.1:6379> lrange list1 0 -1
1) "hello3"
127.0.0.1:6379> lrange list 0 -1
1) "hello"
2) "hello1"
3) "hello2"
# lset 将列表中指定下标的值替换为另外一个值,更新操作
# lset将列表key下标为的值替换成
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 hello
(integer) 1
127.0.0.1:6379> LRANGE list 0 0 
1) "hello"
127.0.0.1:6379> lset list 1 item
(error) ERR index out of range
127.0.0.1:6379> lset list 0 item # 如果存在就会替换当前下标的值
OK
127.0.0.1:6379> LRANGE list 0 0
1) "item"
# linsert 将某个具体的value插入到列把你中某个元素的前面或者后面!
# linsert before | after 的后面插入插入值
127.0.0.1:6379> Rpush mylist "hello" 
(integer) 1 
127.0.0.1:6379> Rpush mylist "world" 
(integer) 2 
127.0.0.1:6379> LINSERT mylist before "world" "other" 
(integer) 3 
127.0.0.1:6379> LRANGE mylist 0 -1 
1) "hello" 
2) "other" 
3) "world" 
127.0.0.1:6379> LINSERT mylist after "world" "new" 
(integer) 4 
127.0.0.1:6379> LRANGE mylist 0 -1 
1) "hello" 
2) "other" 
3) "world" 
4) "new"

7、Set

sadd <key> <value1> <value2> <value3>将一个或多个member元素加入到集合key中,已经存在的member元素将被忽略
smembers<key> 取出该集合的所有值
sismember <key><value>判断集合<key>是否含有该<value>值,有1,没有0
scard<key> 返回该集合的元素个数
srem<key><value1><value2>...删除集合中的某个元素。
spop<key>随机从该集合中吐出一个值。
srandmember<key><n>随机从该集合中取出n个值,不会从集合中删除
smove <source><distination> value把集合中一个值从一个集合移动到另一个集合
sinter <key1><key2>返回两个集合的交集元素
sunion<key1><key2>返回两个元素的并集元素
sdiff<key1><key2>返回两个集合的差集元素(key1中的,不包含key2中的)
127.0.0.1:6379> sadd myset "hello" # set集合中添加匀速 
(integer) 1 
127.0.0.1:6379> sadd myset "kuangshen" 
(integer) 1 
127.0.0.1:6379> sadd myset "lovekuangshen" 
(integer) 1 
127.0.0.1:6379> SMEMBERS myset # 查看指定set的所有值 
1) "hello" 
2) "lovekuangshen" 
3) "kuangshen" 
127.0.0.1:6379> SISMEMBER myset hello # 判断某一个值是不是在set集合中! 
(integer) 1 
127.0.0.1:6379> SISMEMBER myset world 
(integer) 0
127.0.0.1:6379> scard myset # 获取set集合中的内容元素个数! 
(integer) 4
# rem
127.0.0.1:6379> srem myset hello # 移除set集合中的指定元素 
(integer) 1 
127.0.0.1:6379> scard myset 
(integer) 3 
127.0.0.1:6379> SMEMBERS myset 
1) "lovekuangshen2" 
2) "lovekuangshen" 
3) "kuangshen"
# set 无序不重复集合。抽随机! 
127.0.0.1:6379> SMEMBERS myset 
1) "lovekuangshen2" 
2) "lovekuangshen" 
3) "kuangshen" 
127.0.0.1:6379> SRANDMEMBER myset # 随机抽选出一个元素 
"kuangshen" 
127.0.0.1:6379> SRANDMEMBER myset 
"kuangshen" 
127.0.0.1:6379> SRANDMEMBER myset 
"kuangshen" 
127.0.0.1:6379> SRANDMEMBER myset 
"kuangshen" 
127.0.0.1:6379> SRANDMEMBER myset 2 # 随机抽选出指定个数的元素
1) "lovekuangshen" 
2) "lovekuangshen2" 
127.0.0.1:6379> SRANDMEMBER myset 2 
1) "lovekuangshen" 
2) "lovekuangshen2" 
127.0.0.1:6379> SRANDMEMBER myset # 随机抽选出一个元素 
"lovekuangshen2"
# 删除定的key,随机删除key! 
127.0.0.1:6379> SMEMBERS myset 
1) "lovekuangshen2" 
2) "lovekuangshen" 
3) "kuangshen" 
127.0.0.1:6379> spop myset # 随机删除一些set集合中的元素! 
"lovekuangshen2" 
127.0.0.1:6379> spop myset "lovekuangshen" 
127.0.0.1:6379> SMEMBERS myset 
1) "kuangshen"
# 将一个指定的值,移动到另外一个set集合! 
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 myset "kuangshen" 
(integer) 1 
127.0.0.1:6379> sadd myset2 "set2" 
(integer) 1 
127.0.0.1:6379> smove myset myset2 "kuangshen" # 将一个指定的值,移动到另外一个set集合!
(integer) 1 
127.0.0.1:6379> SMEMBERS myset 
1) "world" 
2) "hello" 
127.0.0.1:6379> SMEMBERS myset2 
1) "kuangshen" 
2) "set2"
# (并集) 数字集合类: 
# - 差集 SDIFF 
# - 交集 SINTER
# - 并集 SUNION
127.0.0.1:6379> SDIFF key1 key2 # 差集 
1) "b" 
2) "a" 
127.0.0.1:6379> SINTER key1 key2 # 交集 共同好友就可以这样实现 
1) "c" 
127.0.0.1:6379> SUNION key1 key2 # 并集 
1) "b" 
2) "c" 
3) "e" 
4) "a"
5) "d"

8、Hash

hset <key><field><value><key>集合的filed赋值i<value>
hget <key1><filed> 从key1>集合<filed>取出<value>
hmset <key1><field><value1><key2><field><value2>批量设置hash的值
hexists<key1><field> 查看哈希表key中,给定域filed是否存在。
hkeys <key> 列出该hash集合的所有field
hvals <key>列出该hash集合的所有的value
hincrby <key><field><increment>为哈希表key中的域field的值加上增量 1 -1
hsetnx <key><field><value> 将哈希表key中的域field的值设置为value,当且仅当field不存在
127.0.0.1:6379> hset myhash field1 kuangshen # set一个具体 key-vlaue 
(integer) 1 
127.0.0.1:6379> hget myhash field1 # 获取一个字段值 
"kuangshen" 
127.0.0.1:6379> hmset myhash field1 hello field2 world # set多个 key-vlaue 
OK
127.0.0.1:6379> hmget myhash field1 field2 # 获取多个字段值 
1) "hello" 
2) "world" 
127.0.0.1:6379> hgetall myhash # 获取全部的数据, 
1) "field1" 
2) "hello" 
3) "field2" 
4) "world" 
127.0.0.1:6379> hdel myhash field1 # 删除hash指定key字段!对应的value值也就消失了! 
(integer) 1 
127.0.0.1:6379> hgetall myhash 
1) "field2" 
2) "world"
# hlen
127.0.0.1:6379> hmset myhash field1 hello field2 world 
OK
127.0.0.1:6379> HGETALL myhash 
1) "field2" 
2) "world" 
3) "field1" 
4) "hello" 
127.0.0.1:6379> hlen myhash # 获取hash表的字段数量! 
(integer) 2
127.0.0.1:6379> HEXISTS myhash field1 # 判断hash中指定字段是否存在! 
(integer) 1 
127.0.0.1:6379> HEXISTS myhash field3 
(integer) 0
# 只获得所有field 
# 只获得所有value 
127.0.0.1:6379> hkeys myhash # 只获得所有field 
1) "field2" 
2) "field1"
127.0.0.1:6379> hvals myhash # 只获得所有value 
1) "world" 
2) "hello"
# incr decr 
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 hello # 如果不存在则可以设置 
(integer) 1 
127.0.0.1:6379> hsetnx myhash field4 world # 如果存在则不能设置 
(integer) 0

9、Zset(有序集合)

在set的基础上,增加了一个值,set k1 v1 zset k1 score1 v1

zadd <key><score1><value1><score2><value2> 将一个或多个member元素及其score值加入到有序集key中
zrange <key><start><stop>[WITHSCORES]
返回有序集合key中,下表在<start><stop>之间的元素,带WITHSCORES,可以让分数一起和值返回到结果集。
zrangebyscore key minmax[withscores][limit offset count]
返回有序集合key中,所有score值介于min和max之间(包括min和max)的成员。按score值递增次序排列
zrevrangebyscore key maxmin[withscores][limit offset count] 同上改为从大到小排列
zincrby <key><increment><value> 为元素的score加上增量
zrem <key><value> 删除该集合下的,指定值的元素
zcount <key><min><max> 统计该集合,分数区间内的元素个数
zrank <key><value> 返回该值在集合中的排名,从0开始。
127.0.0.1:6379> zadd myset 1 one # 添加一个值 
(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 kaungshen 
(integer) 1 
# ZRANGEBYSCORE key min max 
127.0.0.1:6379> ZRANGEBYSCORE salary -inf +inf # 显示全部的用户,从小到大! 
1) "kaungshen" 
2) "xiaohong" 
3) "zhangsan" 
127.0.0.1:6379> ZREVRANGE salary 0 -1 # 从大到进行排序! 
1) "zhangsan" 
2) "xiaohong"
3) "kaungshen" 
127.0.0.1:6379> ZRANGEBYSCORE salary -inf +inf withscores # 显示全部的用户并且附带成 绩
1) "kaungshen" 
2) "500" 
3) "xiaohong" 
4) "2500" 
5) "zhangsan" 
6) "5000" 
127.0.0.1:6379> ZRANGEBYSCORE salary -inf 2500 withscores # 显示工资小于2500员工的升序排序! 
1) "kaungshen" 
2) "500" 
3) "xiaohong" 
4) "2500"
# 移除rem中的元素
127.0.0.1:6379> zrange salary 0 -1 
1) "kaungshen" 
2) "xiaohong" 
3) "zhangsan" 
127.0.0.1:6379> zrem salary xiaohong # 移除有序集合中的指定元素 
(integer) 1 
127.0.0.1:6379> zrange salary 0 -1 
1) "kaungshen" 
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 kuangshen 
(integer) 2 
127.0.0.1:6379> zcount myset 1 3 # 获取指定区间的成员数量! 
(integer) 3 
127.0.0.1:6379> zcount myset 1 2 
(integer) 2

10、Geospatial地理位置

地理位置查询:http://www.jsons.cn/lngcode/

# getadd 添加地理位置 
# 规则:两级无法直接添加,我们一般会下载城市数据,直接通过java程序一次性导入! 
# 有效的经度从-180度到180度。 
# 有效的纬度从-85.05112878度到85.05112878度。 
# 当坐标位置超出上述指定范围时,该命令将会返回一个错误。 
127.0.0.1:6379> geoadd china:city 39.90 116.40 beijin 
(error) ERR invalid longitude,latitude pair 39.900000,116.400000
127.0.0.1:6379> geoadd china:city 116.40 39.90 beijing 
(integer) 1 
127.0.0.1:6379> geoadd china:city 121.47 31.23 shanghai 
(integer) 1 
127.0.0.1:6379> geoadd china:city 106.50 29.53 chongqi 114.05 22.52 shengzhen 
(integer) 2 
127.0.0.1:6379> geoadd china:city 120.16 30.24 hangzhou 108.96 34.26 xian 
(integer) 2
# getpos获得当前定位:一定是一个坐标值!
127.0.0.1:6379> GEOPOS china:city beijing # 获取指定的城市的经度和纬度! 
1) 1) "116.39999896287918091" 
   2) "39.90000009167092543" 
127.0.0.1:6379> GEOPOS china:city beijing chongqi 
1) 1) "116.39999896287918091" 
   2) "39.90000009167092543" 
2) 1) "106.49999767541885376" 
   2) "29.52999957900659211"
# GEODIST 两人之间的距离!
# 单位:
# m 表示单位为米。
# km 表示单位为千米。
# mi 表示单位为英里。
# ft 表示单位为英尺。
127.0.0.1:6379> GEODIST china:city beijing shanghai km # 查看上海到北京的直线距离 
"1067.3788" 
127.0.0.1:6379> GEODIST china:city beijing chongqi km # 查看重庆到北京的直线距离 
"1464.0708"
# georadius 以给定的经纬度为中心, 找出某一半径内的元素
127.0.0.1:6379> GEORADIUS china:city 110 30 1000 km 
# 以110,30 这个经纬度为中心,寻找方圆1000km内的城市 
1) "chongqi" 
2) "xian" 
3) "shengzhen" 
4) "hangzhou" 
127.0.0.1:6379> GEORADIUS china:city 110 30 500 km 
1) "chongqi" 
2) "xian" 
127.0.0.1:6379> GEORADIUS china:city 110 30 500 km withdist # 显示到中间距离的位置 
1) 1) "chongqi" 
   2) "341.9374" 
2) 1) "xian" 
   2) "483.8340" 
127.0.0.1:6379> GEORADIUS china:city 110 30 500 km withcoord # 显示他人的定位信息 
1) 1) "chongqi" 
   2) 1) "106.49999767541885376" 
      2) "29.52999957900659211" 
2) 1) "xian" 
   2) 1) "108.96000176668167114" 
      2) "34.25999964418929977" 
127.0.0.1:6379> GEORADIUS china:city 110 30 500 km withdist withcoord count 1 
# 筛选出指定的结果! 
1) 1) "chongqi" 
   2) "341.9374" 
   3) 1) "106.49999767541885376" 
      2) "29.52999957900659211" 
127.0.0.1:6379> GEORADIUS china:city 110 30 500 km withdist withcoord count 2 
1) 1) "chongqi" 
   2) "341.9374" 
   3) 1) "106.49999767541885376" 
      2) "29.52999957900659211" 
2) 1) "xian" 
   2) "483.8340" 
   3) 1) "108.96000176668167114" 
      2) "34.25999964418929977"
# GEORADIUSBYMEMBER 找出位于指定元素周围的其他元素! 
127.0.0.1:6379> GEORADIUSBYMEMBER china:city beijing 1000 km 
1) "beijing" 
2) "xian" 
127.0.0.1:6379> GEORADIUSBYMEMBER china:city shanghai 400 km 
1) "hangzhou" 
2) "shanghai"
# GEOHASH 命令 - 返回一个或多个位置元素的 Geohash 表示
# 该命令将返回11个字符的Geohash字符串!
# 将二维的经纬度转换为一维的字符串,如果两个字符串越接近,那么则距离越近! 
127.0.0.1:6379> geohash china:city beijing chongqi 
1) "wx4fbxxfke0" 
2) "wm5xzrybty0"
# GEO 底层的实现原理其实就是 Zset!我们可以使用Zset命令来操作geo!
127.0.0.1:6379> ZRANGE china:city 0 -1 # 查看地图中全部的元素 
1) "chongqi" 
2) "xian" 
3) "shengzhen" 
4) "hangzhou" 
5) "shanghai" 
6) "beijing" 
127.0.0.1:6379> zrem china:city beijing # 移除指定元素! 
(integer) 1 
127.0.0.1:6379> ZRANGE china:city 0 -1 
1) "chongqi" 
2) "xian" 
3) "shengzhen" 
4) "hangzhou" 
5) "shanghai"

11、Hyperloglog

127.0.0.1:6379> PFadd mykey a b c d e f g h i j # 创建第一组元素 mykey 
(integer) 1 
127.0.0.1:6379> PFCOUNT mykey # 统计 mykey 元素的基数数量 
(integer) 10 
127.0.0.1:6379> PFadd mykey2 i j z x c v b n m # 创建第二组元素 mykey2 
(integer) 1 
127.0.0.1:6379> PFCOUNT mykey2 
(integer) 9 
127.0.0.1:6379> PFMERGE mykey3 mykey mykey2 # 合并两组 mykey mykey2 => mykey3 并集 
OK
127.0.0.1:6379> PFCOUNT mykey3 # 看并集的数量! 
(integer) 15

12、Bitmap

127.0.0.1:6379> setbit sign 0 1
127.0.0.1:6379> setbit sign 1 0
127.0.0.1:6379> setbit sign 2 0
127.0.0.1:6379> setbit sign 3 1
127.0.0.1:6379> setbit sign 4 1
127.0.0.1:6379> setbit sign 5 0
127.0.0.1:6379> setbit sign 6 0
127.0.0.1:6379> getbit sign 3 
(integer) 1 
127.0.0.1:6379> getbit sign 6 
(integer) 0
127.0.0.1:6379> bitcount sign
(integer) 3

13、事务

# 正常执行事务
127.0.0.1:6379> multi # 开启事务 
OK
# 命令入队 
127.0.0.1:6379> set k1 v1
QUEUED 
127.0.0.1:6379> set k2 v2 
QUEUED 
127.0.0.1:6379> get k2 
QUEUED 
127.0.0.1:6379> set k3 v3 
QUEUED 
127.0.0.1:6379> exec # 执行事务 
1) OK 
2) OK 
3) "v2" 
4) OK
# 放弃事务
127.0.0.1:6379> multi # 开启事务 
OK
127.0.0.1:6379> set k1 v1 
QUEUED 
127.0.0.1:6379> set k2 v2 
QUEUED 
127.0.0.1:6379> set k4 v4 
QUEUED 
127.0.0.1:6379> DISCARD # 取消事务 
OK
127.0.0.1:6379> get k4 # 事务队列中命令都不会被执行! 
(nil)
# 编译型异常(代码有问题! 命令有错!) ,事务中所有的命令都不会被执行!
127.0.0.1:6379> multi 
OK
127.0.0.1:6379> set k1 v1 
QUEUED 
127.0.0.1:6379> set k2 v2 
QUEUED 
127.0.0.1:6379> set k3 v3 
QUEUED 
127.0.0.1:6379> getset k3 # 错误的命令 
(error) ERR wrong number of arguments for 'getset' command 
127.0.0.1:6379> set k4 v4 
QUEUED 
127.0.0.1:6379> set k5 v5 
QUEUED 
127.0.0.1:6379> exec # 执行事务报错! 
(error) EXECABORT Transaction discarded because of previous errors. 
127.0.0.1:6379> get k5 # 所有的命令都不会被执行! 
(nil)
# 运行时异常(1/0), 如果事务队列中存在语法性,那么执行命令的时候,其他命令是可以正常执行的,错误命令抛出# 异常!
127.0.0.1:6379> set k1 "v1" 
OK
127.0.0.1:6379> multi 
OK
127.0.0.1:6379> incr k1 # 会执行的时候失败! 
QUEUED 
127.0.0.1:6379> set k2 v2 
QUEUED 
127.0.0.1:6379> set k3 v3 
QUEUED 
127.0.0.1:6379> get k3 
QUEUED 
127.0.0.1:6379> exec 
1) (error) ERR value is not an integer or out of range # 虽然第一条命令报错了,但是 依旧正常执行成功了! 
2) OK 
3) OK 
4) "v3" 
127.0.0.1:6379> get k2 
"v2" 
127.0.0.1:6379> get k3 
"v3"
# 监控! Watch, 使用watch 可以当做redis的乐观锁操作
127.0.0.1:6379> set money 100 
OK
127.0.0.1:6379> set out 0 
OK
127.0.0.1:6379> watch money # 监视 money 对象 
OK
127.0.0.1:6379> multi # 事务正常结束,数据期间没有发生变动,这个时候就正常执行成功! 
OK
127.0.0.1:6379> DECRBY money 20 
QUEUED 
127.0.0.1:6379> INCRBY out 20 
QUEUED 
127.0.0.1:6379> exec 
1) (integer) 80 
2) (integer) 20
127.0.0.1:6379> watch money # 监视 money 
OK
127.0.0.1:6379> multi 
OK
127.0.0.1:6379> DECRBY money 10 
QUEUED 
127.0.0.1:6379> INCRBY out 10 
QUEUED 
127.0.0.1:6379> exec # 执行之前,另外一个线程,修改了我们的值,这个时候,就会导致事务执行失败!
(nil)
# 如果修改失败,获取最新的值就好
# 如果发现事务执行失败,就先解锁
127.0.0.1:6379> UNWATCH
# 获取最新的值,再次监视,select version
127.0.0.1:6379> watch money # 监视 money 
OK
127.0.0.1:6379> multi 
OK
127.0.0.1:6379> DECRBY money 10 
QUEUED 
127.0.0.1:6379> INCRBY out 10 
QUEUED 
127.0.0.1:6379> exec
1) (integer) 60 
2) (integer) 40

14、Redis密码设置

127.0.0.1:6379> ping 
PONG 
127.0.0.1:6379> config get requirepass # 获取redis的密码 
1) "requirepass" 
2) "" 
127.0.0.1:6379> config set requirepass "123456" # 设置redis的密码 
OK
127.0.0.1:6379> config get requirepass # 发现所有的命令都没有权限了 
(error) NOAUTH Authentication required. 
127.0.0.1:6379> ping 
(error) NOAUTH Authentication required. 
127.0.0.1:6379> auth 123456 # 使用密码进行登录! 
OK
127.0.0.1:6379> config get requirepass 
1) "requirepass" 
2) "123456"

15、持久化

# 查看需要存在的位置
127.0.0.1:6379> config get dir 
1) "dir" 
2) "/usr/local/bin" # 如果在这个目录下存在 dump.rdb 文件,启动就会自动恢复其中的数据

rdb触发机制:

1、save的规则满足的情况下,会自动触发rdb规则。

2、执行 flushall 命令,也会触发我们的rdb规则。

3、退出redis,也会产生 rdb 文件。

rdb 触发后会自动生成一个 dump.rdb。

如果 aof 持久化文件有错位,这时候 redis 是启动不起来的,我们需要修复这个aof文件,redis 给我们提供了一个

工具 redis-check-aof --fix

$ redis-check-aof --fix appendonly.aof

16、发布订阅

# 订阅端:
127.0.0.1:6379> SUBSCRIBE kuangshenshuo # 订阅一个频道 kuangshenshuo 
Reading messages... (press Ctrl-C to quit) 
1) "subscribe" 
2) "kuangshenshuo" 
3) (integer) 1 
# 等待读取推送的信息 
1) "message" # 消息 
2) "kuangshenshuo" # 那个频道的消息 
3) "hello,kuangshen" # 消息的具体内容 
1) "message" 
2) "kuangshenshuo" 
3) "hello,redis"
# 发送端:
127.0.0.1:6379> PUBLISH kuangshenshuo "hello,kuangshen" # 发布者发布消息到频道! 
(integer) 1 
127.0.0.1:6379> PUBLISH kuangshenshuo "hello,redis" # 发布者发布消息到频道! 
(integer) 1 127.0.0.1:6379>

17、分布式锁

实现分布式锁之前先看两个 Redis 命令:

17.1 SETNX

将key设置值为value,如果key不存在,这种情况下等同SET命令。 当key存在时,什么也不做。

返回值:

  • 1 如果key被设置了
  • 0 如果key没有被设置

例子:

redis> SETNX mykey "Hello"
(integer) 1
redis> SETNX mykey "Hello"
(integer) 0
redis> GET mykey
"Hello"
redis> 

17.2 GETSET

自动将key对应到value并且返回原来key对应的value。如果key存在但是对应的value不是字符串,就返回错误。

返回值:

  • 返回之前的旧值,如果之前Key不存在将返回nil

例子:

redis> INCR mycounter
(integer) 1
redis> GETSET mycounter "0"
"1"
redis> GET mycounter
"0"
redis>

这两个命令在 java 中对应为 setIfAbsentgetAndSet

17.3 分布式锁Java的实现

redisTemplate.opsForValue().setIfAbsent(key, value);
redisTemplate.opsForValue().getAndSet(key, value);
redisTemplate.opsForValue().getOperations().delete(key);

你可能感兴趣的:(redis,redis)