Redis生产环境批量的key需要删除,如何优雅实现且不影响线上环境?有密码认证的如何实现批量删除?有哪些实现方法?
在 Redis 命令行中删除匹配某个模式的 key,最推荐 的方式是使用 redis-cli --scan 配合管道,避免阻塞服务。
推荐方式:非阻塞删除
bash
# 匹配并删除(使用 xargs)
redis-cli --scan --pattern "user:*" | xargs redis-cli del
# 如果 key 包含空格,使用 -0 参数安全处理
redis-cli --scan --pattern "user:*" | xargs -0 redis-cli del
# Redis 4.0+ 建议用 UNLINK 代替 DEL(异步删除,不阻塞)
redis-cli --scan --pattern "user:*" | xargs redis-cli unlink
交互式方式:SCAN + DEL
如果在 redis-cli 交互模式中,或需要更精确控制:
bash
# 使用 SCAN 渐进式遍历(不会阻塞 Redis)
127.0.0.1:6379> SCAN 0 MATCH user:* COUNT 100
1) "17" # 下次游标
2) 1) "user:1001"
2) "user:1002"
3) "user:1003"
# 然后对返回的 key 逐个删除
127.0.0.1:6379> DEL user:1001 user:1002 user:1003
脚本批量删除(生产环境推荐)
对于大量 key,建议写个小脚本循环 SCAN:
bash
#!/bin/bash
# 安全删除匹配 key 的脚本
cursor=0
while true; do
result=$(redis-cli SCAN $cursor MATCH "user:*" COUNT 100)
cursor=$(echo "$result" | head -n 1)
keys=$(echo "$result" | tail -n +2)
if [ -n "$keys" ]; then
echo "$keys" | xargs redis-cli unlink
fi
[ "$cursor" == "0" ] && break
done
⚠️ 避免使用 KEYS + DEL
bash
# 以下命令仅适用于测试环境/数据量极小的情况
# 生产环境绝对不要这样做!会阻塞整个 Redis 实例
redis-cli KEYS "user:*" | xargs redis-cli del # ❌ 危险
对比总结
| 方式 | 是否阻塞 | 适用场景 |
|---|---|---|
KEYS + DEL |
✅ 严重阻塞 | 仅测试环境,key 数量 < 1000 |
SCAN + DEL/UNLINK |
❌ 不阻塞 | 生产环境标准做法 |
redis-cli --scan |
❌ 不阻塞 | 命令行快速操作 |
UNLINK |
❌ 异步删除 | Redis 4.0+ 大量删除首选 |
核心原则 :生产环境永远优先使用 SCAN 或 UNLINK,避免 KEYS 和 DEL 大量 key。
如果你已经在 redis-cli 交互模式里,key 数量只有 100 个左右,可以直接这样操作:
最简单的方式:先查再删
bash
# 1. 列出匹配的 key
127.0.0.1:6379> keys user:*
# 2. 复制返回的 key 列表,直接执行 del
127.0.0.1:6379> del user:001 user:002 user:003 ...
100 个 key 手动复制粘贴虽然麻烦,这是最稳妥的方式。
如果想在交互模式里批量处理
Redis 交互模式本身不支持管道或循环,但你可以:
方式一:退出交互模式,用 shell 一句话搞定
bash
# 先 quit 退出 redis-cli,然后在 bash 里执行
redis-cli keys "user:*" | xargs redis-cli del
方式二:用 EVAL 执行 Lua 脚本(无需退出)
bash
127.0.0.1:6379> EVAL "return redis.call('del', unpack(redis.call('keys', ARGV[1])))" 0 user:*
⚠️ 注意:
unpack在 key 数量很多时会报 "too many results" 错误,1000 个一般没问题。
方式三:分步 SCAN + DEL(最安全)
bash
127.0.0.1:6379> scan 0 match user:* count 100
1) "0"
2) 1) "user:001"
2) "user:002"
...
# 如果游标返回 0,说明遍历完了,然后手动 del
127.0.0.1:6379> del user:001 user:002 ...