Redis 批量删除键

Redis 批量删除键

1. 简介

在Redis中批量优雅的删除大量键是一个很麻烦的问题,下面例举常用的方法和优缺点。

2. keys+del命令

使用keys使用查找所有匹配的键,然后在使用del一起批量删除。

优点

  1. 操作简单。

缺点

  1. 查找性能特别差,需要遍历所有键。如果redis key特别多,还有可能造成redis短时间无法提供服务。
  2. 网络性能占用多,所有匹配键都需要发送到redis客户端,如果匹配键特别多会占用大量网络带宽(这个问题可以通过Lua脚本来解决)。
  3. 占用内存多,需要大量内存暂存已经找到的键。
py 复制代码
import redis

r = redis.Redis(host='127.0.0.1', port=6379, db=0)

# 查找
keys = r.keys("example-*")
# 删除
r.delete(*keys)

3. SCAN命令

  1. 使用SCAN命令迭代匹配特定模式的键。
  2. 对于每次SCAN迭代返回的键,使用DEL命令进行删除。
  3. 重复上述步骤,直到SCAN命令返回的游标为0,表示迭代完成。

优点

  1. 非阻塞,不会影响Redis服务器的其他操作。
  2. 可以控制每次迭代的键数量,减少内存占用。

缺点

  1. 在高并发场景下,很难保证所有匹配的键都完全删除,需要多次迭代。
py 复制代码
import redis

r = redis.Redis(host='127.0.0.1', port=6379, db=0)

cursor = 0  # 以0开始
while True:
    cursor, keys = r.scan(cursor=cursor, match='example-*', count=100)
    if cursor == 0:  # 以0结束
        break

    if keys:
        r.delete(*keys)

4. UNLINK命令

UNLINK命令是Redis提供的一个异步删除命令,与DEL命令不同,UNLINK命令会立即返回,而删除操作将在后台异步进行。

  1. 使用SCAN命令或KEYS命令获取要删除的键列表。
  2. 使用UNLINK命令代替DEL命令进行删除。

优点

  1. 异步删除,不会阻塞Redis服务器。

缺点

  1. 删除的完成时间不可预测。
  2. 在某些情况下可能需要额外的逻辑,保证在需要删除的键已经删除。
py 复制代码
import redis

r = redis.Redis(host='127.0.0.1', port=6379, db=0)

cursor = 0  # 以0开始
while True:
    cursor, keys = r.scan(cursor=cursor, match='example-*', count=100)
    if cursor == 0:  # 以0结束
        break

    if keys:
        r.unlink(*keys)

5. Lua脚本

上面所有的示例都可以转换成Lua脚本版本,下面只演示其中一个。

优点

  1. 大量降低redis客户端redis服务器的RTT时间。
  2. 大量节约了redis服务器的带宽。

缺点

  1. 提高操作的复杂。
py 复制代码
import redis

r = redis.Redis(host='127.0.0.1', port=6379, db=0)

lua_script = """
local key1 = KEYS[1]
local delete_count = 0
local delete_keys = redis.call('keys', key1)
if #delete_keys > 0 then
    delete_count = redis.call('del',unpack(delete_keys))
end

return delete_count
"""

# 注册脚本
script = r.register_script(lua_script)

# 执行
result = script(keys=['example-*'])
print(result)

6. 总结

  1. 尽量避免在生产环境使用keys :使用keys不可避免的会出现性能问题。
  2. 分批次删除 :不要一次删除防止redis服务端处理大量键进入阻塞状态。
相关推荐
爱的叹息7 小时前
Java 连接 Redis 的驱动(Jedis、Lettuce、Redisson、Spring Data Redis)分类及对比
java·redis·spring
松韬8 小时前
Spring + Redisson:从 0 到 1 搭建高可用分布式缓存系统
java·redis·分布式·spring·缓存
天上掉下来个程小白8 小时前
Redis-14.在Java中操作Redis-Spring Data Redis使用方式-操作列表类型的数据
java·redis·spring·springboot·苍穹外卖
·云扬·8 小时前
深度剖析 MySQL 与 Redis 缓存一致性:理论、方案与实战
redis·mysql·缓存
汤姆大聪明9 小时前
Redisson 操作 Redis Stream 消息队列详解及实战案例
redis·spring·缓存·maven
csjane10791 天前
Redis原理:rename命令
java·redis
爱学测试的雨果1 天前
Postman —— postman实现参数化
软件测试·功能测试·测试工具·lua·postman
Chandler241 天前
Redis:内存淘汰原则,缓存击穿,缓存穿透,缓存雪崩
数据库·redis·缓存
❀͜͡傀儡师1 天前
多台服务器上docker部署 Redis 集群
运维·服务器·redis
Foyo Designer1 天前
【 <二> 丹方改良:Spring 时代的 JavaWeb】之 Spring Boot 中的国际化:支持多语言的 RESTful API
java·spring boot·redis·后端·spring·缓存·restful