Redis非关系形数据库

  Redis 是开源的,免费的,高性能的 非关系形数据库。

数据存储在内存,以key-value形式存储,类似于我们学习的字典 {name: ‘jack’ }

可以设置过期时间,过期自动删除,也可以做持久化

非关系型数据库: Redis 在非关系型数据库中没有表这个概念,也就没有表关系这样的概念,都是k/v 格式的。

**关系型数据库(有表结构): MySQL

Redis 优势:

  • 性能极高 – Redis读的速度是11w次/s,写的速度是8.1w次/s ,都是基于内存的,读写速度比较快。

  • 丰富的数据类型 – Redis支持 Strings, Lists, Hashes, Sets 及 Ordered Sets 数据类型操作。

  • 原子性 – Redis的所有操作都是原子性的,要么成功执行,要么失败不执行。支持简单的事务,即原子性,通过MULTI和EXEC指令包起来。如下所有的操作全部执行或者全部不执行

>multi
>set name jack
>set age 23
>set sex male
>exec

  • 丰富的特性 – Redis还支持 publish/subscribe, 通知, key 过期等等特性。

Redis的缺点:

  • 基于内存型数据库, 主机断电时数据全部丢失。

  • key同时过期会造成Redis数据库的“雪崩”,容易把整个服务搞垮。

redis五大数据类

  1. String 字符串类型

  2. List 列表

  3. Hash 哈希 理解成 python的字典 xiaoming = {“name”:"xiaoming"}

  4. Set 无序列表 [zhangsan,lisi ,wangwu]

  5. ZSet 有序列表 [zhangsan :60,lisi:100 ,wangwu:48] 默认从小到大

Redis通用命令

redis默认有16个 数据库(0-15) select 3 切换到数据库3

info 查看redis服务的信息 info replication查看主从复制的状况

keys * 查看所有的key, 生产环境中不使用 可能阻塞redis

查看key的数据类型 type key1

key 是否存在 exists key1

删除一个key del key1

重命名 rename key key1

清空数据库 flushdb/flushall

Redis写法

String类型

# 增
set key value # 设置一个key-value 映射 value可以是字符串或者数字
mset key1 value1 key2 value2 ... # 同时设置多个key-value映射 
set key value ex 60 # 数据有效期60秒
set key value nx # 只有当key不存在时,才会存入

# 查
get key # 查看一个key的值
mget key1 key2 key3 ... # 同时获取多个key
strlen key1 # 获取key的长度
strrange key 0 4 # 获取字符串[0-4]之间的字符

# 改
set # set 命令也可以用来修改数据
setrange key 1 oooo # 修改字符串的一部分:从1的位置开始修改字符串,以oooo覆盖

# string的特殊类型:数字
# 数字可以直接增减
incr key # 自增 1
decr key # 自减 1
incrby key 5 # 自增 5
decrby key 5 # 自减 5

# 删
del name

Set类型(无序列表,无序集合)

Set 是 String 类型无序的容器结构。集合成员是唯一的,不能出现重复的数据。

Redis 中集合是通过哈希表实现的,所以添加,删除,查找的复杂度都是 O(1)。

特点:无序的,元素不可重复的

# 增
sadd room zhangsan lisi wangwu # 增加3个玩家到游戏中

# 查
smember room # 查看当前所有在线的玩家
scard room # 查看当前房间里有多少玩家
sismember room zhangsan # 判断zhangsan是否已经加入游戏

# 改
srem room zhangsan lisi # 把zhangsan , lisi 从房间移除

# 删除
del name

Zset类型

它是 String 类型元素的有序集合,每个元素唯一且都会关联一个 double 类型的分值,根据分值来为集合中的成员从小到大排序。

# 增
zadd guns 30 ak47 47 dp28 100 m247 


# 查
zrange guns 0 -1 [withscores]  # 查看guns 下面的所有枪械,按照从小到大排列
zrevrange guns 0 -1 [withscores] # 从大到小排列
zrange guns (1 3  # 查看guns (1-3]的所有枪械

zrangebyscore guns (30 100 # 查看弹容量在(30-100]之间的所有枪械
zrangebyscore guns 30 100 limit 2 1 withscores # 分页,每页1条,第三页
zrevrangebyscore guns 100 30 # 查看弹容量在(30-100]之间的所有枪械
zrevrangebyscore guns 100 30  limit 2 1 withscores # 分页,每页1条,第三页


zcard guns # 查看系统中枪的种类数
zcount guns (20 50 # 获取(20-50] 区间内的所有枪械数

zrank name dp28 # 按照从小到大,查看排名
zrevrank name dp28 # 按照从大到小

zscore guns ak47 # 查看ak47的弹容量

# 改
zincrby guns 10 m247 # 给m247枪增加10发子弹
zincrby guns -10 dp28 # 给dp28 减少10发子弹

zrem guns ak47 ... # 按照字符串来移除元素,可同时多个
zremrangebyrank guns 1 2 # 按照弹容量排名  0-3 区间内的元素移除
zremrangebyscore guns 0 30 # 把弹容量小于30发的,全部删除掉

# 删
del guns

Hash类型

哈希加密 关系不大

hash类型 大大降低了查找时候的难度 O(1)

ak47 -> 弹容量 威力

ak47 = {
    "name":"ak47",
    "bullet":30,
    "bullet_now":27,
    "power":50,
    "num":1,
}
世间万物,全部转化成这种形式

角色 = {
    性别:男、女
    攻击力:
    防御力:
    技能
    等级
}

Redis hash 是一个 string 类型的 field(字段) 和 value(值) 的映射表,hash 特别适合用于存储字典数据。

# 增
hset ak47 num 1 bullet 30 now_bullet 27 power 50 # 设置一个ak47Hash字典,分别是 数量 弹容量 当前子弹  威力


# 查
hget ak47 bullet # 获取ak47的弹容量

hkeys ak47 # 获取ak47的所有属性
hvals ak47 # 获取ak47的所有值
hgetall ak47 # 获取ak47下面的所有字段和值


# 改
hdel ak47 num # 删除一个或多个字段

# 删
del guns

List类型

# 增
lpush history maoyi liangtuo leifengmao # 从list左边依次插入,顺序要注意
rpush history bijini # 从列表右边依次插入

# 查
lrange history 0 -1 # 查看所有
lrange history 0 4 # 查看下标为[0-4]的元素

# 改
lset history 2 qiuyi # 修改指定下标的元素值
lpop history # 从左边删除一个元素
rpop history # 从右边删除一个元素
lrem history 5 chenyi # 删除name中值为zhangsan的元素 最多删除5个,0为删除所有
ltrim history 1 5 # 只保留数组中[1-5]的元素,把其他的删除掉

# 删除
del history
brpop history 5 # 当list为空时,删除整个LIST,并阻塞redis进程5秒中
blpop history 5 # 删除整个list 并阻塞redis进程5秒中

Python写法常用

import redis

# 连接数据库
r = redis.Redis(host='127.0.0.1', port=6379,db=0)
# 使用连接池连接数据库。这样就可以实现多个Redis实例共享一个连接池
pool = redis.ConnectionPool(host='127.0.0.1', port=6379)
r = redis.Redis(connection_pool=pool)

# =============================1、String 操作===============================

#在Redis中设置值,默认不存在则创建,存在则修改
r.set('name', 'zhangsan')
r.mset(name1='zhangsan', name2='lisi')  #批量设置值
r.mset({"name3":'zhangsan1', "name4":'lisi1'})   #批量设置值
# 字符串局部更新。setrange(name, offset, value)
r.setrange("name",1,"z")   #修改字符串内容,从指定字符串索引开始向后替换
r.setrange("name",6,"zzzzzzz")   #如果新值太长时,则向后添加


# 参数:set(name, value, ex=None, px=None, nx=False, xx=False)
#      ex,过期时间(秒)
#      px,过期时间(毫秒)
#      nx,如果设置为True,则只有name不存在时,当前set操作才执行,同setnx(name, value)
#      xx,如果设置为True,则只有name存在时,当前set操作才执行

# r.setex(name, value, time)
# #设置过期时间(秒)
#
# r.psetex(name, time_ms, value)
# #设置过期时间(豪秒)

# 字符串转化为整型,再自增属性mount对应的值,当属性mount不存在时,则创建mount=amount,否则,则自增,amount为自增数(整数)
print(r.incr("mount",amount=2))
print(r.incr("mount"))
print(r.incr("mount",amount=3))
print(r.incr("mount",amount=6))
print(r.get("mount")) #输出:12
# 字符串转化为浮点数,再自增。incrbyfloat(self, name, amount=1.0)
print(r.incrbyfloat('mount', amount=1.0))
#自减name对应的值,当name不存在时,则创建name=amount,否则,则自减,amount为自增数(整数)
print(r.decr("mount",amount=6))

#在name对应的值后面追加内容
r.set("name","zhangsan")
r.append("name","lisi")
print(r.get("name"))    #输出:zhangsanlisi

# 获取值
print(r.get('name')) # 获取值
print(r.mget("name1","name2"))   #批量获取
print(r.mget(["name3","name4"]))   #批量获取

# 读取后重设  getset(name, value)
#设置新值,打印原值
print(r.getset("name1","wangwu")) #输出:zhangsan
print(r.get("name1")) #输出:wangwu

#根据字节获取子序列  getrange(key, start, end)
print(r.getrange("name",0,3))#输出:zzan

print(r.strlen("name")) #返回name对应值的字节长度(一个汉字3个字节)





# =============================2、Hash 操作===============================
# 增改操作
# hset(name, key, value) #name对应的hash中设置一个键值对(不存在,则创建,否则,修改)
r.hset("dic_name","a1","aa")
r.hmset("dic_name",{"a1":"aa","b1":"bb"})   #在name对应的hash中批量设置键值对,mapping:字典

#自增hash中key对应的值,不存在则创建key=amount(amount为整数)
print(r.hincrby("dic_name","a",amount=2))
#自增hash中key对应的值,不存在则创建key=amount(amount为浮点数)
print(r.hincrbyfloat("dic_name","a",amount=1.0))

# 查询操作
print(r.hget("dic_name","a1"))  #在name对应的hash中根据key获取value
print(r.hmget("dic_name",["a1","b1"]))     # 在name对应的hash中获取多个key的值
print(r.hmget("dic_name","a1","b1"))  # 在name对应的hash中获取多个key的值
print(r.hgetall("dic_name"))  #获取name对应hash的所有键值
print(r.hlen("dic_name"))  #hlen(name) 获取hash中键值对的个数
print(r.hkeys("dic_name"))  #hkeys(name) 获取hash中所有的key的值
print(r.hvals("dic_name"))  #hvals(name) 获取hash中所有的value的值

print(r.hexists("dic_name","a1")) # 检查name对应的hash是否存在当前传入的key


# 删除
r.hdel("dic_name","a1")  #删除指定name对应的key所在的键值对

# =============================3、List 操作===============================
#增改操作
# 在name对应的list中添加元素,每个新的元素都添加到列表的最左边.不存在列表是创建列表
r.lpush("list_name",2)
r.lpush("list_name",3,4,5)#保存在列表中的顺序为5,4,3,2
# rpush(name,values)#同lpush,但每个新的元素都添加到列表的最右边
# lpushx(name,value)#在name对应的list中添加元素,只有name已经存在时,值添加到列表的最左边
# rpushx(name,value)#在name对应的list中添加元素,只有name已经存在时,值添加到列表的最右边

# linsert(name, where, refvalue, value))  其中where: BEFORE(前)或AFTER(后),refvalue: 列表内的值 value: 要插入的数据
r.linsert("list_name","BEFORE","2","SS") #在列表内找到第一个元素2,在它前面插入SS
r.lset("list_name",0,"bbb") #对list中的某一个索引位置重新赋值
r.lpop("list_name")  #移除列表的左侧第一个元素,返回值则是第一个元素
r.rpoplpush('list1_name', 'list2_name')  # 从一个列表取出最右边的元素,同时将其添加至另一个列表的最左边

# 查询操作
print(r.llen("list_name"))  # name对应的list元素的个数
print(r.lindex("list_name",1))  #根据索引获取列表内元素
print(r.lrange("list_name",0,-1))  #分片获取元素


# 删除操作
r.lrem("list_name","SS",num=0)   #r.lrem(name, value, num)  其中value: 要删除的值   num:   num=0 删除列表中所有的指定值;num=2 从前到后,删除2个;num=-2 从后向前,删除2个'''
r.ltrim("list_name",0,2)  #移除列表内没有在该索引之内的值

print(r.blpop(["list_name","list_name1"],timeout=0))  #将多个列表排列,按照从左到右去移除各个列表内的元素。timeout: 超时时间,获取完所有列表的元素之后,阻塞等待列表内有数据的时间(秒), 0 表示永远阻塞
# r.brpop(keys, timeout)  #同blpop,将多个列表排列,按照从右向左去移除各个列表内的元素

# =============================4、set 操作===============================
# 增改操作
#给name对应的集合中添加元素
r.sadd("set_name","aa")
r.sadd("set_name","aa","bb")
r.smove("set_name1", "set_name2", 'aa')  #将某个元素从一个集合中移动到另外一个集合
r.spop("set_name")  #从集合的右侧移除一个元素,并将其返回



# 查询操作
print(r.smembers('set_name'))   #获取name对应的集合的所有成员
print(r.scard("set_name")) #获取name对应的集合中的元素个数
print(r.sdiff("set_name","set_name1","set_name2"))# 在第一个name对应的集合中且不在其他name对应的集合的元素集合
# sdiffstore(dest, keys, *args) #相当于把sdiff获取的值加入到dest对应的集合中

print(r.sinter("set_name","set_name1","set_name2")) # 获取多个name对应集合的交集
# sinterstore(dest, keys, *args) #获取多个name对应集合的交集,再将其加入到dest对应的集合中
print(r.sunion("set_name","set_name1","set_name2"))  #获取多个name对应的集合的并集
# sunionstore(dest,keys, *args)#获取多个name对应的集合的并集,并将结果保存到dest对应的集合中

print(r.srandmember("set_name2",2))  # 从name对应的集合中随机获取numbers个元素

print(r.sismember("set_name", 'aa'))  #检查value是否是name对应的集合内的元素

# 删除操作
r.srem("set_name2","bb","dd")  #删除name对应的集合中的某些值

# =============================5、有序集合 操作===============================
# 在集合的基础上,为每元素排序,元素的排序需要根据另外一个值来进行比较,所以,对于有序集合,每一个元素有两个值,即:值和分数,分数专门用来做排序。
# 增改操作
r.zadd("zset_name", "a1", 6, "a2", 2,"a3",5)   # 在name对应的有序集合中添加元素
r.zadd('zset_name1', b1=10, b2=5)  # 在name对应的有序集合中添加元素



# 查询操作
print(r.zcard("zset_name"))  #获取有序集合内元素的数量
print(r.zcount("zset_name",1,5))  #获取有序集合中分数在[min,max]之间的个数

r.zincrby("zset_name","a1",amount=2)#自增zset_name对应的有序集合里a1对应的分数

# 按照索引范围获取name对应的有序集合的元素.start起始索引,end终点索引,desc排序规则,默认按照分数从小到大排序,withscores是否获取元素的分数,默认只获取元素的值,score_cast_func 对分数进行数据转换的函数
aa=r.zrange("zset_name",0,1,desc=False,withscores=True,score_cast_func=int)


print(r.zscore("zset_name","value1")) #获取name对应有序集合中 value 对应的分数
print(r.zrank("zset_name", "value1"))  #获取value值在name对应的有序集合中的排行位置(从0开始)
print(r.zrevrank("zset_name", "value1"))#从大到小排序

# 删除操作

r.zrem("zset_name","value1","value2") #删除name对应的有序集合中值是values的成员
r.zremrangebyrank("zset_name", 3, 5)  #根据排行范围删除
r.zremrangebyscore("zset_name", 3, 5) #根据分数范围删除
r.zinterstore("zset_dest",("zset_name1","zset_name2"),aggregate="MAX")  # 获取两个有序集合的交集并放入dest集合,如果遇到相同值不同分数,则按照aggregate进行操作:aggregate的值为: SUM  MIN  MAX
# r.zunionstore(dest, keys, aggregate=None)  #获取两个有序集合的并集并放入dest集合,其他同zinterstore,


# =============================6、通用操作===============================
r.delete('name')  #根据name删除redis中的任意数据类型
print(r.exists('name'))  #检测redis的name是否存在
print(r.keys(pattern='*'))  #根据* ?等通配符匹配获取redis的name
print(r.expire('name' ,time=3000))  # 为某个name设置超时时间
r.rename('name', 'name1')  # 重命名
r.move('name', 'db1')  # 将redis的某个值移动到指定的db下
print(r.randomkey())  #随机获取一个redis的name(不删除)
print(r.type('name'))  # 获取name对应值的类型

# redis-py默认在执行每次请求都会创建(连接池申请连接)和断开(归还连接池)一次连接操作,
# 如果想要在一次请求中指定多个命令,则可以使用pipline实现一次请求指定多个命令,并且默认情况下一次pipline 是原子性操作。
pipe = r.pipeline(transaction=True)

r.set('name', 'zhangsan')
r.set('name', 'lisi')

pipe.execute()

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值