redis c->buf=3的极限测试

本文探讨了在RedisClient中将回复静态缓冲区buf设置为3时,导致命令解析错误的原因。通过分析发现,当buf大小为3时,特定命令会被组织成多个链表结点,在发送过程中可能出现数据读取混乱,进而引发解析命令出错。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

今天做了一个极限的测试,把redisClient的回复静态缓冲区buf,设置为3。这样做的目的是使用c->reply作为回复缓冲。测试的时候出现了解析命令出错的问题。


仔细思考了一下原因如下:当c->buf大小为3时,replconf ack 1会被组织成10个链表结点:



而发送时是一个节点一个节点的发送的。如先发*3\r\n,再发$8\r\n,这时被发送方会读取数据,有可能读到*3\r\n,也有可能读到*3\r\n$8\r\n,所以会出现解析命令出错。

以下是一些可能的代码优化建议: 1. 建议使用 RAII(Resource Acquisition Is Initialization)技术来管理 Redis 连接,以确保在函数退出时正确释放连接。可以使用 C++11 中的 std::unique_ptr 来管理 redisContext 对象。 2. 建议使用 Redis 的 SCAN 命令来遍历所有的 key,而不是使用 KEYS 命令。因为 KEYS 命令在 Redis 中是一个非常耗费资源的命令,而 SCAN 命令则是一种更加高效的方式来遍历所有的 key。 3. 建议在函数开始时使用 redisAppendCommand 函数来批量执行 Redis 命令,而不是在循环中使用 redisCommand 函数执行单个命令。这样可以减少网络往返次数,提高程序的性能。 4. 建议将 QSet<QString> firstLayer 和 QJsonObject keyInfos 改为成员变量,以避免在每次函数调用时重新创建和初始化。 5. 建议使用 C++11 中的 auto 关键字来自动推断变量类型,以简化代码。 下面是应用上述优化后的代码示例: ``` #include "redismanager.h" #include <QDebug> #include <memory> RedisManager::RedisManager(QObject* parent) : QObject(parent) { context = std::unique_ptr<redisContext, decltype(&redisFree)>(redisConnect("127.0.0.1", 6379), redisFree); if (context == NULL || context->err) { if (context) { qDebug() << "Error: " << context->errstr; } else { qDebug() << "Can&#39;t allocate redis context"; } } } void RedisManager::getFirstLayerKeys() { auto firstLayer = QSet<QString>(); auto keyInfos = QJsonObject(); auto cursor = "0"; auto cmds = QStringList(); do { cmds << QString("SCAN %1").arg(cursor); auto reply = std::unique_ptr<redisReply, decltype(&freeReplyObject)>(nullptr, freeReplyObject); if (redisAppendCommand(context.get(), cmds.last().toStdString().c_str()) != REDIS_OK || redisGetReply(context.get(), (void**)&reply) != REDIS_OK || reply == nullptr || reply->type != REDIS_REPLY_ARRAY || reply->elements != 2) { qDebug() << "Failed to execute command."; return; } cursor = QString(reply->element[0]->str); auto keysReply = reply->element[1]; for (size_t i = 0; i < keysReply->elements; i++) { auto key = QString(keysReply->element[i]->str); auto pos = key.indexOf(&#39;:&#39;); if (pos != -1) { auto firstLayerKey = key.left(pos); if (!firstLayer.contains(firstLayerKey)) { firstLayer.insert(firstLayerKey); cmds << QString("KEYS %1:*").arg(firstLayerKey); } } } } while (cursor != "0"); auto reply = std::unique_ptr<redisReply, decltype(&freeReplyObject)>(nullptr, freeReplyObject); for (auto cmd : cmds) { if (redisAppendCommand(context.get(), cmd.toStdString().c_str()) != REDIS_OK) { qDebug() << "Failed to execute command."; return; } } for (size_t i = 0; i < cmds.size(); i++) { if (redisGetReply(context.get(), (void**)&reply) != REDIS_OK || reply == nullptr) { qDebug() << "Failed to execute command."; return; } auto firstLayerKey = cmds[i].mid(4, cmds[i].indexOf(&#39;:&#39;) - 4); if (reply->type == REDIS_REPLY_ARRAY) { keyInfos.insert(firstLayerKey, QJsonValue((int)reply->elements)); } freeReplyObject(reply); } emit firstLayerKeysReady(keyInfos); } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值