pubsub channels 命令返回当前至少有一个订阅者的活跃频道,不保留历史记录;频道非键空间成员,故 keys/scan 无法查到;支持 pattern 过滤,结果需手动 decode。

用 PUBSUB CHANNELS 命令直接查活跃频道
Redis 不会保留“历史频道”,只返回当前至少有一个订阅者的频道——也就是真正“活跃”的频道。直接执行 PUBSUB CHANNELS 就能拿到结果,不需要额外状态维护或扫描键空间。
- 在
redis-cli中运行:PUBSUB CHANNELS,输出类似1) "news.sport" 2) "chat.room:1024" - Python 中推荐用
r.execute_command('PUBSUB CHANNELS'),而不是r.pubsub_channels()(后者是某些旧版 redis-py 的非标准封装,行为不稳定) - 注意:该命令时间复杂度为
O(N),N 是当前活跃频道数;如果频道数上万且调用频繁,可能轻微影响主线程响应(但一般场景完全无感)
为什么 KEYS * 或 SCAN 查不到频道?
频道不是 Redis 的 key,它不占用键空间,也不受过期、持久化或内存淘汰策略影响。试图用 KEYS pattern 或 SCAN 搜索频道名,永远为空——这是最常见的误解来源。
- 发布消息到未被订阅的频道,消息直接丢弃,该频道也不会“激活”或被记录
-
PUBSUB CHANNELS的底层依赖的是 Redis 内部的订阅哈希表,和 DB 中的 key 字典完全隔离 - 如果你需要长期跟踪频道使用情况,得自己用
SET或HyperLogLog配合业务逻辑埋点
用 pattern 参数做频道名过滤
当频道命名有规律(比如都带前缀 log: 或 user:),可以用通配符缩小范围,避免返回海量无关频道。
- 命令示例:
PUBSUB CHANNELS "log:*"或PUBSUB CHANNELS "user:123*" - 注意通配符语法和
KEYS一致:*匹配任意字符,?匹配单个字符,[abc]匹配方括号内任一字符 - 如果 pattern 完全不匹配任何活跃频道,返回空列表,不会报错
Python 中拿到结果后记得解码
redis-py 默认返回字节串(bytes),直接打印会看到 b'chat:general' 这样的输出,容易误判为“没数据”或格式异常。
- 正确处理方式:
[c.decode() for c in r.execute_command('PUBSUB CHANNELS')] - 如果频道名含非 UTF-8 字符(极少见),需指定编码,如
c.decode('latin-1') - 别用
str(c)强转,那会得到"b'xxx'"字面量字符串,不是你想要的频道名










