首页 > 数据库 >hiredis的同步模式和异步模式

hiredis的同步模式和异步模式

时间:2023-06-22 16:44:34浏览次数:51  
标签:异步 cout REDIS 模式 hiredis str REPLY reply

1.什么是hiredis

Hiredis 是一个 C 语言编写的 Redis 客户端库,用于与 Redis 数据库进行交互。它提供了一个简洁而高效的接口,使开发人员可以方便地在自己的 C/C++ 项目中使用 Redis。
Hiredis 是一个开源项目,可从其官方 GitHub 仓库获取源代码,并在符合 BSD 许可证的条件下使用和分发。它被广泛应用于各种 C/C++ 项目中,特别是那些需要与 Redis 数据库进行快速、可靠和高性能交互的应用程序。

2.安装hiredis

centos中可以使用yum直接安装:
yum install hiredis-devel

或者直接从源码安装:

git clone https://github.com/redis/hiredis.git
cd hiredis/
make
make install

3.使用hiredis

使用同步的方式连接redis,只需要使用以下几个函数:

redisContext *redisConnect(const char *ip, int port);
void *redisCommand(redisContext *c, const char *format, ...);
void freeReplyObject(void *reply);

其中,发送指令的函数在使用上与printf类似,支持不定参数:

reply = redisCommand(context, "SET foo bar");
reply = redisCommand(context, "SET foo %s", value);
reply = redisCommand(context, "SET key:%s %s", myid, value);

同时也支持二进制的参数:
reply = redisCommand(context, "SET foo %b", value, (size_t) valuelen);
%b表示二进制,然后传入二进制value及其长度valuelen

可以封装成一个C++类:

// g++ hiredis.cpp -lhiredis
#include <hiredis/hiredis.h>
#include <iostream>
#include <string>
using namespace std;

class Redis 
{
public:
        Redis(const char * host, int port = 6379){
                c = redisConnect(host, port);
                if (c == NULL){
                        cout << "cannot allocate redis context" << endl;
                        return;
                }

                if (c->err){
                        cout << c->errstr << endl;
                        c = NULL;
                        return;
                }
        }

        ~Redis(){
                if (c){
                        redisFree(c);
                }
        }

        bool isConnect()
        {
                return c != NULL;
        }

        bool Set(const char * key, const char * value)
        {
                redisReply *reply = (redisReply*) redisCommand(c, "set %s %s", key, value);
                if (reply == NULL){
                        return false;
                }
                bool success = false;
                if (reply->type == REDIS_REPLY_STATUS){
                        cout << reply->str << endl;
                        success = true;
                } else if (reply->type == REDIS_REPLY_ERROR) {
                        cout << reply->str << endl;
                }

                freeReplyObject(reply);
                return success;
        }

        string Get(const char* key)
        {
                redisReply *reply = (redisReply*) redisCommand(c, "get %s", key);
                string value;
                if (reply == NULL){
                        return value;
                }
                if (reply->type == REDIS_REPLY_STRING){
                        value = reply->str;
                }else if (reply->type == REDIS_REPLY_ERROR){
                        cout << reply->str << endl;
                }
                freeReplyObject(reply);
                return value;
        }

        bool HSet(const char * key, const char * field, const char * value)
        {
                redisReply *reply = (redisReply*) redisCommand(c, "hset %s %s %s", key, field, value);
                if (reply == NULL){
                        return false;
                }
                bool success = false;
                if (reply->type == REDIS_REPLY_STATUS){
                        cout << reply->str << endl;
                        success = true;
                } else if (reply->type == REDIS_REPLY_ERROR) {
                    cout << reply->str << endl;
                }
                freeReplyObject(reply);
                return success;
        }

        string HGet(const char* key, const char * field)
        {
                redisReply *reply = (redisReply*) redisCommand(c, "hget %s %s", key, field);
                string value;
                if (reply == NULL){
                        return value;
                }
                if (reply->type == REDIS_REPLY_STRING){
                        value = reply->str;
                }else if (reply->type == REDIS_REPLY_ERROR){
                        cout << reply->str << endl;
                }
                freeReplyObject(reply);
                return value;
        }

private:
        redisContext *c;

};

int main(){
        Redis r("127.0.0.1", 6379);
        if (!r.isConnect()){
                return 0;
        }

        r.Set("id", "100");
        string value = r.Get("id");
        cout << value << endl;

        r.HSet("map", "1", "200");
        cout << r.HGet("map", "1") << endl;

        return 0;
}

4.hiredis的异步模式

使用hiredis的同步模式时,每一个请求都需要等待回应之后,才能进行下一个请求,时间消耗在IO等待上。为了提高效率,可以使用异步模式,同时发送多个请求,然后回调处理。

使用hiredis的异步模式,需要绑定一个事件适配器,常用的有libevent libuv等。
下面以libevent为例:

#include <iostream>
#include <signal.h>
#include <hiredis/async.h>
#include <hiredis/adapters/libevent.h>

struct event_base *eventBase = nullptr;

void signalHandler(int sig, short events, void *arg) {
    printf("Received signal %d, exiting...\n", sig);

    // 停止事件循环
    event_base_loopbreak(eventBase);
}

void commandCallback(redisAsyncContext* context, void* reply, void* data) {
    redisReply* r = reinterpret_cast<redisReply*>(reply);
    if (r == nullptr) {
        std::cout << "Command error: " << context->errstr << std::endl;
        return;
    }

    if (r->type == REDIS_REPLY_ERROR) {
        std::cout << "Command error: " << r->str << std::endl;
        return;
    }

    // 处理命令返回的结果
    std::cout << r->str << std::endl;

    // 这里不需要释放 reply 对象,hiredis的异步机制会自动释放
}

void connectCallback(const redisAsyncContext* context, int status) {
    if (status == REDIS_OK) {
        std::cout << "Connected to Redis" << std::endl;

        // 如果有密码,可以在这里验证
        // ...

        // 在连接成功后发送指令
        redisAsyncContext* asyncContext = const_cast<redisAsyncContext*>(context);
        redisAsyncCommand(asyncContext, commandCallback, nullptr, "SET mykey Hello");
        redisAsyncCommand(asyncContext, commandCallback, nullptr, "GET mykey");
    } else {
        std::cout << "Connection error" << std::endl;
    }
}

int main() {
    eventBase = event_base_new();
    redisAsyncContext* context = redisAsyncConnect("127.0.0.1", 6379);

    if (context->err) {
        std::cout << "Connection error: " << context->errstr << std::endl;
        return 1;
    }

    redisLibeventAttach(context, eventBase);
    redisAsyncSetConnectCallback(context, connectCallback);

    struct event * signalEvent = evsignal_new(eventBase, SIGINT, signalHandler, nullptr);;
    evsignal_add(signalEvent, nullptr);

    std::cout << "开始事件循环..." << std::endl;
    event_base_dispatch(eventBase);
    std::cout << "事件循环结束..." << std::endl;

    event_free(signalEvent);
    event_base_free(eventBase);
    redisAsyncDisconnect(context);

    return 0;
}

参考资料

标签:异步,cout,REDIS,模式,hiredis,str,REPLY,reply
From: https://www.cnblogs.com/lcc9527/p/17497983.html

相关文章

  • 智能化制造:一种新的生产模式
    目录1.引言2.技术原理及概念3.实现步骤与流程4.应用示例与代码实现讲解5.优化与改进6.结论与展望智能化制造:一种新的生产模式随着人工智能技术的不断发展,智能化制造成为了现代制造业的一个重要趋势。智能化制造能够实现生产过程中的自动化、智能化和数据化,提高生产效率和......
  • 20230430 28. 访问者模式 - 男女对比
    介绍访问者模式(Visitor),表示一个作用于某对象结构中的各元素的操作。它使你可以在不改变各元素的类的前提下定义作用于这些元素的新操作访问者模式适用于数据结构相对稳定的系统访问者模式把数据结构和作用于结构上的操作之间的耦合解脱开,使得操作集合可以相对自由地演化。......
  • 20230427 23. 命令模式 - 烧烤点单
    介绍命令模式(Command),将一个请求封装为一个对象,从而使你可用不同的请求对客户进行参数化;对请求排队或记录请求日志,以及支持可撤销的操作Command类,用来声明执行操作的接口ConcreteCommand类,将一个接收者对象绑定于一个动作,调用接收者相应的操作,以实现executeCommandInvoker......
  • 20230428 24. 职责链模式 - 审批流程
    介绍职责链模式(ChainofResponsibility):使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的耦合关系。将这个对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它为止。Handler类,定义一个处理请示的接口ConcreteHandler类,具体处理者类,处理它所负责的请......
  • 20230430 26. 享元模式 - 网站复用
    介绍享元模式(Flyweight),运用共享技术有效地支持大量细粒度的对象。在享元对象内部并且不会随环境改变而改变的共享部分,可以称为是享元对象的内部状态,而随环境改变而改变的、不可以共享的状态就是外部状态了。事实上,享元模式可以避免大量非常相似类的开销。在程序设计中,有时需......
  • 20230430 27. 解释器模式 - 音符
    解释器模式(interpreter),给定一个语言,定义它的文法的一种表示,并定义一个解释器,这个解释器使用该表示来解释语言中的句子。解释器模式需要解决的是,如果一种特定类型的问题发生的频率足够高,那么可能就值得将该问题的各个实例表述为一个简单语言中的句子。这样就可以构建一个解释器......
  • 实战设计模式解耦项目网络层框架
    作者:依乐祝写在前面仔细想想我们的这个极简CMS系统,可以说很简单,简单到都无须进行特殊的架构设计,只需按照你所熟悉的编码方式直接进行快速的编码实现即可,如果做得好的话,访问量上来了你再加一个缓存处理完全能够支撑一定的并发!如下图所示:我们前期先进行单体架构的实现,等后期分布式系......
  • Android 开发之Activity的启动模式-SingleTop
    接下来,介绍下Activity的另一种启动模式-栈顶复用模式(SingleTop)SingleTopsingleTop模式,它的表现几乎和standard模式一模一样,一个singleTopActivity的实例可以无限多,唯一的区别是如果在栈顶已经有一个相同类型的Activity实例,Intent不会再创建一个Activity,而是通过onNewIntent()被......
  • Android-Kotlin-单例模式
    先看一个案例,非单例模式的案例:描述Dog对象:packagecn.kotlin.kotlin_oop08classDog(varname:String,varcolor:String){/***显示狗狗的名字*/funshowDogName(){println("狗狗的名字是:${this.name}")}/***显示狗狗的颜......
  • 如何判断Apache服务器的工作模式
    如何才能知道当前的apache2使用什么工作机制?我们可以通过httpd-l命令列出apache的所有模块,就可以知道其工作方式: prefork工作模式 如果httpd-l列出prefork.c。则表示是prefork工作方式。如下面所示:Compiledinmodules:core.cprefork.chttp_co......