Redis 是一种高性能的键值存储数据库,广泛应用于缓存、排行榜、计数器等场景。在实际业务中,我们经常需要查询符合特定条件的数据,例如找出 value 大于某个阈值(如 10)的键值对。然而,直接遍历所有键并使用 GET 命令逐个检查可能会导致性能问题,尤其是当数据量较大时。
本文将围绕 如何高效查询 Redis 中满足条件的数据 展开讨论,从最初的简单实现到优化后的高效方案,并结合 Java 代码示例,帮助开发者掌握 Redis 数据查询的最佳实践。
假设我们有以下需求:
DB1(-n 1)存储了大量形如 flow:count:1743061930:* 的键。value > 10 的所有键值对,并统计总数。最初的 Shell 脚本如下:
redis-cli -h 10.206.0.16 -p 6379 -n 1 --scan --pattern "flow:count:1743061930:*" | \
while read key; do
value=$(redis-cli -h 10.206.0.16 -p 6379 -n 1 GET "$key")
if [ "$value" != "1" ]; then
echo "$key: $value"
fi
done | tee /dev/stderr | wc -l | awk '{print "Total count: " $1}'该方案的问题:
GET,网络开销大。[ "$value" != "1" ] 是字符串比较,数值比较更合适。tee、wc、awk 多个管道影响性能。优化后的版本:
redis-cli -h 10.206.0.16 -p 6379 -n 1 --scan --pattern "flow:count:1743061930:*" | \
while read key; do
redis-cli -h 10.206.0.16 -p 6379 -n 1 GET "$key"
done | \
awk '$1 > 10 {count++; print} END {print "Total count: " count}'优化点:
value,减少网络开销。awk 进行数值比较:$1 > 10 比 Shell 字符串比较更高效。awk 同时完成过滤、输出和计数。如果仍需保留键名:
redis-cli -h 10.206.0.16 -p 6379 -n 1 --scan --pattern "flow:count:1743061930:*" | \
while read key; do
value=$(redis-cli -h 10.206.0.16 -p 6379 -n 1 GET "$key")
echo "$key: $value"
done | \
awk -F': ' '$2 > 10 {count++; print} END {print "Total count: " count}'Shell 脚本仍然存在多次 GET 的问题,我们可以使用 Redis Pipeline 批量获取数据,减少网络往返时间。
redis-cli -h 10.206.0.16 -p 6379 -n 1 --scan --pattern "flow:count:1743061930:*" | \
xargs -I {} redis-cli -h 10.206.0.16 -p 6379 -n 1 MGET {} | \
awk '$1 > 10 {count++; print} END {print "Total count: " count}'这里使用 xargs + MGET 批量获取 value,减少网络请求次数。
在 Java 应用中,我们可以使用 Jedis 或 Lettuce 客户端优化查询。
import redis.clients.jedis.Jedis;
import redis.clients.jedis.ScanParams;
import redis.clients.jedis.ScanResult;
import java.util.List;
public class RedisValueFilter {
public static void main(String[] args) {
String host = "10.206.0.16";
int port = 6379;
int db = 1;
String pattern = "flow:count:1743061930:*";
int threshold = 10;
try (Jedis jedis = new Jedis(host, port)) {
jedis.select(db);
ScanParams scanParams = new ScanParams().match(pattern).count(100);
String cursor = "0";
int totalCount = 0;
do {
ScanResult<String> scanResult = jedis.scan(cursor, scanParams);
List<String> keys = scanResult.getResult();
cursor = scanResult.getCursor();
// 批量获取 values
List<String> values = jedis.mget(keys.toArray(new String[0]));
// 过滤并统计
for (int i = 0; i < keys.size(); i++) {
String key = keys.get(i);
String valueStr = values.get(i);
if (valueStr != null) {
int value = Integer.parseInt(valueStr);
if (value > threshold) {
System.out.println(key + ": " + value);
totalCount++;
}
}
}
} while (!cursor.equals("0"));
System.out.println("Total count: " + totalCount);
}
}
}优化点:
SCAN 代替 KEYS,避免阻塞 Redis。MGET 批量查询,减少网络开销。Lettuce 是高性能 Redis 客户端,支持异步查询:
import io.lettuce.core.*;
import io.lettuce.core.api.sync.RedisCommands;
import java.util.List;
public class RedisLettuceQuery {
public static void main(String[] args) {
RedisURI uri = RedisURI.create("redis://10.206.0.16:6379/1");
RedisClient client = RedisClient.create(uri);
try (RedisConnection<String, String> connection = client.connect()) {
RedisCommands<String, String> commands = connection.sync();
String pattern = "flow:count:1743061930:*";
int threshold = 10;
int totalCount = 0;
ScanCursor cursor = ScanCursor.INITIAL;
do {
ScanArgs scanArgs = ScanArgs.Builder.matches(pattern).limit(100);
KeyScanCursor<String> scanResult = commands.scan(cursor, scanArgs);
List<String> keys = scanResult.getKeys();
cursor = ScanCursor.of(scanResult.getCursor());
// 批量获取 values
List<KeyValue<String, String>> keyValues = commands.mget(keys.toArray(new String[0]));
for (KeyValue<String, String> kv : keyValues) {
if (kv.hasValue()) {
int value = Integer.parseInt(kv.getValue());
if (value > threshold) {
System.out.println(kv.getKey() + ": " + value);
totalCount++;
}
}
}
} while (!cursor.isFinished());
System.out.println("Total count: " + totalCount);
} finally {
client.shutdown();
}
}
}优势:
RedisReactiveCommands)。方案 | 查询方式 | 网络开销 | 适用场景 |
|---|---|---|---|
原始 Shell | 单 GET 遍历 | 高 | 少量数据 |
优化 Shell + awk | 批量 GET | 中 | 中等数据量 |
Shell + Pipeline | MGET 批量 | 低 | 大数据量 |
Java + Jedis | SCAN + MGET | 低 | 生产环境 |
Java + Lettuce | 异步 SCAN | 最低 | 高并发 |
KEYS 命令:使用 SCAN 替代,防止阻塞 Redis。MGET 或 Pipeline 批量查询。awk 或 Java 直接比较数值,而非字符串。通过优化,我们可以显著提升 Redis 大数据查询的效率,降低服务器负载,适用于高并发生产环境。