基于Hiredis异步API的聊天系统实现
基于Hiredis异步API的聊天系统实现 上一篇文章http://blog.csdn.net/qq_34788352/article/details/51313027使用Hiredis的同步API实现了发送消息的客户端,当我使用同步API实现订阅频道客户端时,一旦订阅频道,就会出现无法操作的情况,这是就是同步和异步的
基于Hiredis异步API的聊天系统实现
上一篇文章http://blog.csdn.net/qq_34788352/article/details/51313027使用Hiredis的同步API实现了发送消息的客户端,当我使用同步API实现订阅频道客户端时,一旦订阅频道,就会出现无法操作的情况,这是就是同步和异步的问题。使用同步API,订阅频道后,客户端会进入阻塞状态,等待订阅频道发布的消息,不能实现既订阅频道,又能发布消息的功能。为了实现一个客户端既能订阅频道,又能发布消息的功能,就需要使用Hiredis的异步API。
首先特别感谢黄天霸、tickTick、vah101,当我遇到各种各样奇怪问题的时候,你们的帖子给与了我解答,谢谢。
开始正题,hiredis异步的实现主要是依靠redis自带的ae事件库或者libev事件库或者libevent的事件库或者libuv事件库。网上一些人是通过libevent事件库来实现,本系统则使用Redis自带ae事件库来实现,Redis不用libevent事件库而选择重写一个ae事件库,必定有其道理。
首先介绍使用到的异步API,位于async.h中:
<code class=" hljs perl">redisAsyncContext <span class="hljs-variable">*redisAsyncConnect</span>(const char <span class="hljs-variable">*ip</span>, <span class="hljs-keyword">int</span> port); <span class="hljs-regexp">//</span>用于建立异步连接 <span class="hljs-keyword">int</span> redisAsyncSetConnectCallback(redisAsyncContext <span class="hljs-variable">*ac</span>, redisConnectCallback <span class="hljs-variable">*fn</span>); <span class="hljs-regexp">//</span>设置连接回调函数,回调函数形式:void callback(const redisAsyncContext <span class="hljs-variable">*c</span>, <span class="hljs-keyword">int</span> status); <span class="hljs-keyword">int</span> redisAsyncSetDisconnectCallback(redisAsyncContext <span class="hljs-variable">*ac</span>, redisDisconnectCallback <span class="hljs-variable">*fn</span>); <span class="hljs-regexp">//</span>设置断开连接回调函数,回调函数形式:void callback(const redisAsyncContext <span class="hljs-variable">*c</span>, <span class="hljs-keyword">int</span> status); void redisAsyncDisconnect(redisAsyncContext <span class="hljs-variable">*ac</span>); <span class="hljs-regexp">//</span>断开异步连接 void redisAsyncFree(redisAsyncContext <span class="hljs-variable">*ac</span>); <span class="hljs-regexp">//</span>释放建立连接时,创建的redisAsyncContext结构 <span class="hljs-keyword">int</span> redisAsyncCommand(redisAsyncContext <span class="hljs-variable">*ac</span>, redisCallbackFn <span class="hljs-variable">*fn</span>, void <span class="hljs-variable">*privdata</span>, const char <span class="hljs-variable">*format</span>, ...); <span class="hljs-regexp">//</span>发送Redis命令,需要实现一个回调函数来出来命令的返回,fn是回调函数的地址,回调函数形式:void callback(redisAsyncContext <span class="hljs-variable">*c</span>, void <span class="hljs-variable">*reply</span>, void <span class="hljs-variable">*pridata</span>); </code>
有了上面的异步API,就可以开始客户端的撰写。
首先封装订阅端
<code class=" hljs cpp"><span class="hljs-comment">//sub_client.h</span> <span class="hljs-preprocessor">#ifndef REDIS_SUB_CLIENT_H </span> <span class="hljs-preprocessor">#define REDIS_SUB_CLIENT_H </span> <span class="hljs-keyword">extern</span> <span class="hljs-string">"C"</span> { <span class="hljs-preprocessor">#include <stdlib.h> </span> <span class="hljs-preprocessor">#include <async.h> </span> <span class="hljs-preprocessor">#include <adapters/ae.h> </span> <span class="hljs-preprocessor">#include <unistd.h> </span> <span class="hljs-preprocessor">#include <pthread.h> </span> <span class="hljs-preprocessor">#include <semaphore.h> </span> } <span class="hljs-preprocessor">#include <string> </span> <span class="hljs-preprocessor">#include <vector> </span> <span class="hljs-keyword">using</span> <span class="hljs-keyword">namespace</span> <span class="hljs-built_in">std</span>; <span class="hljs-keyword">class</span> CRedisSubClient { <span class="hljs-keyword">public</span>: CRedisSubClient(); ~CRedisSubClient(); <span class="hljs-keyword">bool</span> init(); <span class="hljs-comment">//初始化,事件对象,信号量 </span> <span class="hljs-keyword">bool</span> uninit(); <span class="hljs-comment">//释放对象</span> <span class="hljs-keyword">bool</span> connect(); <span class="hljs-comment">//连接服务器</span> <span class="hljs-keyword">bool</span> disconnect(); <span class="hljs-comment">//断开服务器</span> <span class="hljs-comment">//订阅频道 </span> <span class="hljs-keyword">bool</span> subscribe(<span class="hljs-keyword">const</span> <span class="hljs-built_in">string</span> &channel_name); <span class="hljs-keyword">private</span>: <span class="hljs-comment">// 下面三个回调函数供redis服务调用 </span> <span class="hljs-comment">// 连接回调 </span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">void</span> connect_callback(<span class="hljs-keyword">const</span> redisAsyncContext *redis_context, <span class="hljs-keyword">int</span> status); <span class="hljs-comment">// 断开连接的回调 </span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">void</span> disconnect_callback(<span class="hljs-keyword">const</span> redisAsyncContext *redis_context, <span class="hljs-keyword">int</span> status); <span class="hljs-comment">// 执行命令回调 </span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">void</span> command_callback(redisAsyncContext *redis_context, <span class="hljs-keyword">void</span> *reply, <span class="hljs-keyword">void</span> *privdata); <span class="hljs-comment">// 事件分发线程函数 </span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">void</span> *event_thread(<span class="hljs-keyword">void</span> *data); <span class="hljs-keyword">void</span> *event_proc(); <span class="hljs-keyword">private</span>: <span class="hljs-comment">// ae事件对象 </span> aeEventLoop *loop; <span class="hljs-comment">// 事件线程ID </span> pthread_t _event_thread; <span class="hljs-comment">// 事件线程的信号量 </span> sem_t _event_sem; <span class="hljs-comment">// hiredis异步对象 </span> redisAsyncContext *_redis_context; }; <span class="hljs-preprocessor">#endif </span> </code>
使用extern “C”是因为redis和hiredis都是c写的,当使用c++链接c代码生成的库中的函数时,会出现undefined reference的问题。
订阅端的实现
<code class=" hljs cpp"><span class="hljs-comment">//sub_client.cpp</span> <span class="hljs-preprocessor">#include <stddef.h> </span> <span class="hljs-preprocessor">#include <assert.h> </span> <span class="hljs-preprocessor">#include <string.h> </span> <span class="hljs-preprocessor">#include "sub_client.h" </span> <span class="hljs-keyword">using</span> <span class="hljs-keyword">namespace</span> <span class="hljs-built_in">std</span>; CRedisSubClient::CRedisSubClient():loop(<span class="hljs-number">0</span>), _event_thread(<span class="hljs-number">0</span>), _redis_context(<span class="hljs-number">0</span>) { } CRedisSubClient::~CRedisSubClient() { } <span class="hljs-keyword">bool</span> CRedisSubClient::init() { loop = aeCreateEventLoop(<span class="hljs-number">64</span>); <span class="hljs-comment">// 创建ae对象</span> <span class="hljs-keyword">if</span> (NULL == loop) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Create redis event failed.\n"</span>); <span class="hljs-keyword">return</span> <span class="hljs-keyword">false</span>; } <span class="hljs-built_in">memset</span>(&_event_sem, <span class="hljs-number">0</span>, <span class="hljs-keyword">sizeof</span>(_event_sem)); <span class="hljs-keyword">int</span> ret = sem_init(&_event_sem, <span class="hljs-number">0</span>, <span class="hljs-number">0</span>); <span class="hljs-comment">//初始化线程信号量</span> <span class="hljs-keyword">if</span> (ret != <span class="hljs-number">0</span>) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Init sem failed.\n"</span>); <span class="hljs-keyword">return</span> <span class="hljs-keyword">false</span>; } <span class="hljs-keyword">return</span> <span class="hljs-keyword">true</span>; } <span class="hljs-keyword">bool</span> CRedisSubClient::uninit() { loop = NULL; sem_destroy(&_event_sem); <span class="hljs-keyword">return</span> <span class="hljs-keyword">true</span>; } <span class="hljs-keyword">bool</span> CRedisSubClient::connect() { _redis_context = redisAsyncConnect(<span class="hljs-string">"127.0.0.1"</span>, <span class="hljs-number">6379</span>); <span class="hljs-comment">// 异步连接到redis服务器上,使用6380端口</span> <span class="hljs-keyword">if</span> (NULL == _redis_context) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">": Connect redis failed.\n"</span>); <span class="hljs-keyword">return</span> <span class="hljs-keyword">false</span>; } <span class="hljs-keyword">if</span> (_redis_context->err) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Connect redis error: %d, %s\n"</span>, _redis_context->err, _redis_context->errstr); <span class="hljs-comment">// 输出错误信息 </span> <span class="hljs-keyword">return</span> <span class="hljs-keyword">false</span>; } redisAeAttach(loop,_redis_context); <span class="hljs-comment">// 将事件绑定到redis context上,使redis的回调跟事件关联 </span> <span class="hljs-comment">// 创建事件处理线程 </span> <span class="hljs-keyword">int</span> ret = pthread_create(&_event_thread, <span class="hljs-number">0</span>, &CRedisSubscriber::event_thread, <span class="hljs-keyword">this</span>); <span class="hljs-keyword">if</span> (ret != <span class="hljs-number">0</span>) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Create event thread failed.\n"</span>); disconnect(); <span class="hljs-keyword">return</span> <span class="hljs-keyword">false</span>; } <span class="hljs-comment">// 设置连接回调,当异步调用连接后,服务器处理连接请求结束后调用,通知调用者连接的状态 </span> redisAsyncSetConnectCallback(_redis_context, &CRedisSubClient::connect_callback); <span class="hljs-comment">// 设置断开连接回调,当服务器断开连接后,通知调用者连接断开,调用者可以利用这个函数实现重连 </span> redisAsyncSetDisconnectCallback(_redis_context, &CRedisSubClient::disconnect_callback); <span class="hljs-comment">// 启动事件线程 </span> sem_post(&_event_sem); <span class="hljs-keyword">return</span> <span class="hljs-keyword">true</span>; } <span class="hljs-keyword">bool</span> CRedisSubClient::disconnect() { <span class="hljs-keyword">if</span> (_redis_context) { redisAsyncDisconnect(_redis_context); redisAsyncFree(_redis_context); _redis_context = NULL; } <span class="hljs-keyword">return</span> <span class="hljs-keyword">true</span>; } <span class="hljs-keyword">bool</span> CRedisSubClient::subscribe(<span class="hljs-keyword">const</span> <span class="hljs-built_in">string</span> &channel_name) { <span class="hljs-keyword">int</span> ret = redisAsyncCommand(_redis_context, &CRedisSubscriber::command_callback, <span class="hljs-keyword">this</span>, <span class="hljs-string">"SUBSCRIBE %s"</span>, channel_name.c_str()); <span class="hljs-comment">//订阅一个频道</span> <span class="hljs-keyword">if</span> (REDIS_ERR == ret) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Subscribe command failed: %d\n"</span>, ret); <span class="hljs-keyword">return</span> <span class="hljs-keyword">false</span>; } <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Subscribe success: %s\n"</span>, channel_name.c_str()); <span class="hljs-keyword">return</span> <span class="hljs-keyword">true</span>; } <span class="hljs-keyword">void</span> CRedisSubClient::connect_callback(<span class="hljs-keyword">const</span> redisAsyncContext *redis_context, <span class="hljs-keyword">int</span> status) { <span class="hljs-keyword">if</span> (status != REDIS_OK) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Error: %s\n"</span>, redis_context->errstr); } <span class="hljs-keyword">else</span> { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Redis connected!"</span>); } } <span class="hljs-keyword">void</span> CRedisSubClient::disconnect_callback( <span class="hljs-keyword">const</span> redisAsyncContext *redis_context, <span class="hljs-keyword">int</span> status) { <span class="hljs-keyword">if</span> (status != REDIS_OK) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">": Error: %s\n"</span>, redis_context->errstr); } } <span class="hljs-comment">// 消息接收回调函数 </span> <span class="hljs-keyword">void</span> CRedisSubClient::command_callback(redisAsyncContext *redis_context, <span class="hljs-keyword">void</span> *reply, <span class="hljs-keyword">void</span> *privdata) { <span class="hljs-keyword">if</span> (NULL == reply || NULL == privdata) { <span class="hljs-keyword">return</span> ; } redisReply *redis_reply = <span class="hljs-keyword">reinterpret_cast</span><redisReply *>(reply); <span class="hljs-comment">// 订阅接收到的消息是一个带三元素的数组 </span> <span class="hljs-keyword">if</span> (redis_reply->type == REDIS_REPLY_ARRAY && redis_reply->elements == <span class="hljs-number">3</span>) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Recieve message:%s %s %s\n"</span>, redis_reply->element[<span class="hljs-number">0</span>]->str redis_reply->element[<span class="hljs-number">1</span>]->str redis_reply->element[<span class="hljs-number">2</span>]->str); } } <span class="hljs-keyword">void</span> *CRedisSubClient::event_thread(<span class="hljs-keyword">void</span> *data) { <span class="hljs-keyword">if</span> (NULL == data) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">": Error!\n"</span>); assert(<span class="hljs-keyword">false</span>); <span class="hljs-keyword">return</span> NULL; } CRedisSubClient *self_this = <span class="hljs-keyword">reinterpret_cast</span><CRedisSubClient *>(data); <span class="hljs-keyword">return</span> self_this->event_proc(); } <span class="hljs-keyword">void</span> *CRedisSubClient::event_proc() { sem_wait(&_event_sem); <span class="hljs-comment">//进行事件处理循环 </span> aeMain(loop); <span class="hljs-keyword">return</span> NULL; } </code>
发布端的封装:
<code class=" hljs cpp"><span class="hljs-comment">//pub_client.h</span> <span class="hljs-preprocessor">#ifndef REDIS_PUB_CLIENT_H </span> <span class="hljs-preprocessor">#define REDIS_PUB_CLIENT_H </span> <span class="hljs-keyword">extern</span> <span class="hljs-string">"C"</span> { <span class="hljs-preprocessor">#include <stdlib.h> </span> <span class="hljs-preprocessor">#include <async.h> </span> <span class="hljs-preprocessor">#include <adapters/ae.h> </span> <span class="hljs-preprocessor">#include <unistd.h> </span> <span class="hljs-preprocessor">#include <pthread.h> </span> <span class="hljs-preprocessor">#include <semaphore.h> </span> } <span class="hljs-preprocessor">#include <string> </span> <span class="hljs-preprocessor">#include <vector> </span> <span class="hljs-keyword">using</span> <span class="hljs-keyword">namespace</span> <span class="hljs-built_in">std</span>; <span class="hljs-keyword">class</span> CRedisPubClient { <span class="hljs-keyword">public</span>: CRedisPubClient(); ~CRedisPubClient(); <span class="hljs-keyword">bool</span> init(); <span class="hljs-comment">//初始化,事件对象,信号量 </span> <span class="hljs-keyword">bool</span> uninit(); <span class="hljs-comment">//释放对象</span> <span class="hljs-keyword">bool</span> connect(); <span class="hljs-comment">//连接服务器</span> <span class="hljs-keyword">bool</span> disconnect(); <span class="hljs-comment">//断开服务器</span> <span class="hljs-comment">//订阅频道 </span> <span class="hljs-keyword">bool</span> publish(<span class="hljs-keyword">const</span> <span class="hljs-built_in">string</span> &channel_name, <span class="hljs-keyword">const</span> <span class="hljs-built_in">string</span> &message); <span class="hljs-keyword">private</span>: <span class="hljs-comment">// 下面三个回调函数供redis服务调用 </span> <span class="hljs-comment">// 连接回调 </span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">void</span> connect_callback(<span class="hljs-keyword">const</span> redisAsyncContext *redis_context, <span class="hljs-keyword">int</span> status); <span class="hljs-comment">// 断开连接的回调 </span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">void</span> disconnect_callback(<span class="hljs-keyword">const</span> redisAsyncContext *redis_context, <span class="hljs-keyword">int</span> status); <span class="hljs-comment">// 执行命令回调 </span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">void</span> command_callback(redisAsyncContext *redis_context, <span class="hljs-keyword">void</span> *reply, <span class="hljs-keyword">void</span> *privdata); <span class="hljs-comment">// 事件分发线程函数 </span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">void</span> *event_thread(<span class="hljs-keyword">void</span> *data); <span class="hljs-keyword">void</span> *event_proc(); <span class="hljs-keyword">private</span>: <span class="hljs-comment">// ae事件对象 </span> aeEventLoop *loop; <span class="hljs-comment">// 事件线程ID </span> pthread_t _event_thread; <span class="hljs-comment">// 事件线程的信号量 </span> sem_t _event_sem; <span class="hljs-comment">// hiredis异步对象 </span> redisAsyncContext *_redis_context; }; <span class="hljs-preprocessor">#endif </span></code>
发布端的实现:
<code class=" hljs cpp"><span class="hljs-comment">//pub_client.cpp</span> <span class="hljs-preprocessor">#include <stddef.h> </span> <span class="hljs-preprocessor">#include <assert.h> </span> <span class="hljs-preprocessor">#include <string.h> </span> <span class="hljs-preprocessor">#include "pub_client.h" </span> <span class="hljs-keyword">using</span> <span class="hljs-keyword">namespace</span> <span class="hljs-built_in">std</span>; CRedisPubClient::CRedisPubClient():loop(<span class="hljs-number">0</span>), _event_thread(<span class="hljs-number">0</span>), _redis_context(<span class="hljs-number">0</span>) { } CRedisPubClient::~CRedisPubClient() { } <span class="hljs-keyword">bool</span> CRedisPubClient::init() { loop = aeCreateEventLoop(<span class="hljs-number">64</span>); <span class="hljs-comment">// 创建ae对象</span> <span class="hljs-keyword">if</span> (NULL == loop) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Create redis event failed.\n"</span>); <span class="hljs-keyword">return</span> <span class="hljs-keyword">false</span>; } <span class="hljs-built_in">memset</span>(&_event_sem, <span class="hljs-number">0</span>, <span class="hljs-keyword">sizeof</span>(_event_sem)); <span class="hljs-keyword">int</span> ret = sem_init(&_event_sem, <span class="hljs-number">0</span>, <span class="hljs-number">0</span>); <span class="hljs-comment">//初始化线程信号量</span> <span class="hljs-keyword">if</span> (ret != <span class="hljs-number">0</span>) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Init sem failed.\n"</span>); <span class="hljs-keyword">return</span> <span class="hljs-keyword">false</span>; } <span class="hljs-keyword">return</span> <span class="hljs-keyword">true</span>; } <span class="hljs-keyword">bool</span> CRedisPubClient::uninit() { loop = NULL; sem_destroy(&_event_sem); <span class="hljs-keyword">return</span> <span class="hljs-keyword">true</span>; } <span class="hljs-keyword">bool</span> CRedisPubClient::connect() { _redis_context = redisAsyncConnect(<span class="hljs-string">"127.0.0.1"</span>, <span class="hljs-number">6379</span>); <span class="hljs-comment">// 异步连接到redis服务器上,使用6380端口</span> <span class="hljs-keyword">if</span> (NULL == _redis_context) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">": Connect redis failed.\n"</span>); <span class="hljs-keyword">return</span> <span class="hljs-keyword">false</span>; } <span class="hljs-keyword">if</span> (_redis_context->err) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Connect redis error: %d, %s\n"</span>, _redis_context->err, _redis_context->errstr); <span class="hljs-comment">// 输出错误信息 </span> <span class="hljs-keyword">return</span> <span class="hljs-keyword">false</span>; } redisAeAttach(loop,_redis_context); <span class="hljs-comment">// 将事件绑定到redis context上,使redis的回调跟事件关联 </span> <span class="hljs-comment">// 创建事件处理线程 </span> <span class="hljs-keyword">int</span> ret = pthread_create(&_event_thread, <span class="hljs-number">0</span>, &CRedisSubscriber::event_thread, <span class="hljs-keyword">this</span>); <span class="hljs-keyword">if</span> (ret != <span class="hljs-number">0</span>) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Create event thread failed.\n"</span>); disconnect(); <span class="hljs-keyword">return</span> <span class="hljs-keyword">false</span>; } <span class="hljs-comment">// 设置连接回调,当异步调用连接后,服务器处理连接请求结束后调用,通知调用者连接的状态 </span> redisAsyncSetConnectCallback(_redis_context, &CRedisSubClient::connect_callback); <span class="hljs-comment">// 设置断开连接回调,当服务器断开连接后,通知调用者连接断开,调用者可以利用这个函数实现重连 </span> redisAsyncSetDisconnectCallback(_redis_context, &CRedisSubClient::disconnect_callback); <span class="hljs-comment">// 启动事件线程 </span> sem_post(&_event_sem); <span class="hljs-keyword">return</span> <span class="hljs-keyword">true</span>; } <span class="hljs-keyword">bool</span> CRedisPubClient::disconnect() { <span class="hljs-keyword">if</span> (_redis_context) { redisAsyncDisconnect(_redis_context); redisAsyncFree(_redis_context); _redis_context = NULL; } <span class="hljs-keyword">return</span> <span class="hljs-keyword">true</span>; } <span class="hljs-keyword">bool</span> CRedisPubClient::publish(<span class="hljs-keyword">const</span> <span class="hljs-built_in">string</span> &channel_name, <span class="hljs-keyword">const</span> <span class="hljs-built_in">string</span> &message) { <span class="hljs-keyword">int</span> ret = redisAsyncCommand(_redis_context, &CRedisPublisher::command_callback, <span class="hljs-keyword">this</span>, <span class="hljs-string">"PUBLISH %s %s"</span>, channel_name.c_str(), message.c_str()); <span class="hljs-comment">//发布消息 </span> <span class="hljs-keyword">if</span> (REDIS_ERR == ret) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Publish command failed: %d\n"</span>, ret); <span class="hljs-keyword">return</span> <span class="hljs-keyword">false</span>; } <span class="hljs-keyword">return</span> <span class="hljs-keyword">true</span>; } <span class="hljs-keyword">void</span> CRedisPubClient::connect_callback(<span class="hljs-keyword">const</span> redisAsyncContext *redis_context, <span class="hljs-keyword">int</span> status) { <span class="hljs-keyword">if</span> (status != REDIS_OK) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Error: %s\n"</span>, redis_context->errstr); } <span class="hljs-keyword">else</span> { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Redis connected!"</span>); } } <span class="hljs-keyword">void</span> CRedisPubClient::disconnect_callback( <span class="hljs-keyword">const</span> redisAsyncContext *redis_context, <span class="hljs-keyword">int</span> status) { <span class="hljs-keyword">if</span> (status != REDIS_OK) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">": Error: %s\n"</span>, redis_context->errstr); } } <span class="hljs-comment">// 消息接收回调函数 </span> <span class="hljs-keyword">void</span> CRedisPubClient::command_callback(redisAsyncContext *redis_context, <span class="hljs-keyword">void</span> *reply, <span class="hljs-keyword">void</span> *privdata) { <span class="hljs-keyword">if</span> (NULL == reply || NULL == privdata) { <span class="hljs-keyword">return</span> ; } redisReply *redis_reply = <span class="hljs-keyword">reinterpret_cast</span><redisReply *>(reply); <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Publish: %s"</span>,redis_reply.str); } <span class="hljs-keyword">void</span> *CRedisPubClient::event_thread(<span class="hljs-keyword">void</span> *data) { <span class="hljs-keyword">if</span> (NULL == data) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">": Error!\n"</span>); assert(<span class="hljs-keyword">false</span>); <span class="hljs-keyword">return</span> NULL; } CRedisPubClient *self_this = <span class="hljs-keyword">reinterpret_cast</span><CRedisPubClient *>(data); <span class="hljs-keyword">return</span> self_this->event_proc(); } <span class="hljs-keyword">void</span> *CRedisPubClient::event_proc() { sem_wait(&_event_sem); <span class="hljs-comment">//进行事件处理循环 </span> aeMain(loop); <span class="hljs-keyword">return</span> NULL; } </code>
测试封装的sub_client和pub_client类:
<code class=" hljs cpp"><span class="hljs-comment">//test_subpub.cpp</span> <span class="hljs-preprocessor">#include "pub_client.h"</span> <span class="hljs-preprocessor">#include "sub_client.h" </span> <span class="hljs-keyword">int</span> main(<span class="hljs-keyword">int</span> argc, <span class="hljs-keyword">char</span> *argv[]) { CRedisPubClient publisher; CRedisSubClient subcriber; <span class="hljs-keyword">bool</span> ret_pub = publisher.init(); <span class="hljs-keyword">bool</span> ret_sub = subcriber.init(); <span class="hljs-keyword">if</span> (!ret_sub&&!ret_pub) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"Init failed.\n"</span>); <span class="hljs-keyword">return</span> <span class="hljs-number">0</span>; } ret_pub = publisher.connect(); ret_sub = subcriber.connect(); <span class="hljs-keyword">if</span> (!ret_sub&&!ret_pub) { <span class="hljs-built_in">printf</span>(<span class="hljs-string">"connect failed."</span>); <span class="hljs-keyword">return</span> <span class="hljs-number">0</span>; } subscriber.subcribe(<span class="hljs-string">"sports"</span>); <span class="hljs-keyword">while</span> (<span class="hljs-keyword">true</span>) { publisher.publish(<span class="hljs-string">"sports"</span>, <span class="hljs-string">"ball"</span>); sleep(<span class="hljs-number">1</span>); } publisher.disconnect(); publisher.uninit(); subscriber.disconnect(); subscriber.disconnect(); <span class="hljs-keyword">return</span> <span class="hljs-number">0</span>; } </code>
终于到了编译链接运行的阶段,我在这个地方卡了快一天,期间各种编译、链接的错误。直接编译链接会出现找到不aeCreateFileEvent,aeDeleteFileEvent,aeMain等等错误。
- 先将redis/src文件夹中的ae.c,ae.h,ae_epoll.c,config.h,zmalloc.c,zmalloc.h拷贝至hiredis目录下
- 用gcc -c ae.c gcc -c zmalloc.c生成ae.o和zmalloc.o,利用这两个文件生成静态库 ar -r libar.a ae.o zmalloc.o
- 然后编译g++ -o test_subpub test_subpub.cpp pub_client.cpp sub_client.cpp -pthread -I ../ -I./ -I ../adapters ../libhiredis.a ../libae.a
- 运行./test_subpub
可以看到,客户端中消息在不断地滚动,即同时实现了订阅频道和发送消息的功能。

ホットAIツール

Undresser.AI Undress
リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover
写真から衣服を削除するオンライン AI ツール。

Undress AI Tool
脱衣画像を無料で

Clothoff.io
AI衣類リムーバー

Video Face Swap
完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

人気の記事

ホットツール

メモ帳++7.3.1
使いやすく無料のコードエディター

SublimeText3 中国語版
中国語版、とても使いやすい

ゼンドスタジオ 13.0.1
強力な PHP 統合開発環境

ドリームウィーバー CS6
ビジュアル Web 開発ツール

SublimeText3 Mac版
神レベルのコード編集ソフト(SublimeText3)

ホットトピック









General Matrix Multiplication (GEMM) は、多くのアプリケーションやアルゴリズムの重要な部分であり、コンピューター ハードウェアのパフォーマンスを評価するための重要な指標の 1 つでもあります。 GEMM の実装に関する徹底的な調査と最適化は、ハイ パフォーマンス コンピューティングとソフトウェア システムとハードウェア システムの関係をより深く理解するのに役立ちます。コンピューター サイエンスでは、GEMM を効果的に最適化すると、計算速度が向上し、リソースが節約されます。これは、コンピューター システムの全体的なパフォーマンスを向上させるために非常に重要です。 GEMM の動作原理と最適化方法を深く理解することは、最新のコンピューティング ハードウェアの可能性をより有効に活用し、さまざまな複雑なコンピューティング タスクに対してより効率的なソリューションを提供するのに役立ちます。 GEMMのパフォーマンスを最適化することで

7月29日、AITO Wenjieの40万台目の新車のロールオフ式典に、ファーウェイの常務取締役、ターミナルBG会長、スマートカーソリューションBU会長のYu Chengdong氏が出席し、スピーチを行い、Wenjieシリーズモデルの発売を発表した。 8月にHuawei Qiankun ADS 3.0バージョンが発売され、8月から9月にかけて順次アップグレードが行われる予定です。 8月6日に発売されるXiangjie S9には、ファーウェイのADS3.0インテリジェント運転システムが初搭載される。 LiDARの支援により、Huawei Qiankun ADS3.0バージョンはインテリジェント運転機能を大幅に向上させ、エンドツーエンドの統合機能を備え、GOD(一般障害物識別)/PDP(予測)の新しいエンドツーエンドアーキテクチャを採用します。意思決定と制御)、駐車スペースから駐車スペースまでのスマート運転のNCA機能の提供、CAS3.0のアップグレード

Huawei 携帯電話にデュアル WeChat ログインを実装するにはどうすればよいですか?ソーシャルメディアの台頭により、WeChatは人々の日常生活に欠かせないコミュニケーションツールの1つになりました。ただし、多くの人は、同じ携帯電話で同時に複数の WeChat アカウントにログインするという問題に遭遇する可能性があります。 Huawei 社の携帯電話ユーザーにとって、WeChat の二重ログインを実現することは難しくありませんが、この記事では Huawei 社の携帯電話で WeChat の二重ログインを実現する方法を紹介します。まず第一に、ファーウェイの携帯電話に付属するEMUIシステムは、デュアルアプリケーションを開くという非常に便利な機能を提供します。アプリケーションのデュアルオープン機能により、ユーザーは同時に

WeChat は、現在最も人気のあるソーシャル ソフトウェアの 1 つとして、豊富なチャット機能を提供します。ただし、場合によっては、「このチャットを表示しない」状況が発生し、重要な会話が非表示になることがあります。これらのチャットを復元するのは、実際には非常に簡単です。これらの手順に従う限り、非表示のチャットを簡単に復元し、引き続き WeChat がもたらす便利なコミュニケーション エクスペリエンスを楽しむことができます。 WeChat にチャットが表示されない場合、チャットを復元するにはどうすればよいですか? WeChat ではチャット回復方法が表示されない 方法 1: WeChat メッセージ リストでチャット パートナーの名前またはキーワードを直接検索してみてください。検索が見つかった場合は、クリックしてチャット インターフェイスに入り、チャットを復元して表示できます。方法 2、友人チャットを通じて復元: WeChat を開き、アドレス帳をクリックし、非表示チャットに表示されている友人を見つけ、クリックしてメッセージを送信します

Apple 16 システムの最適なバージョンは iOS16.1.4 です。iOS16 システムの最適なバージョンは人によって異なります。日常の使用体験における追加と改善も多くのユーザーから賞賛されています。 Apple 16 システムの最適なバージョンはどれですか? 回答: iOS16.1.4 iOS 16 システムの最適なバージョンは人によって異なる場合があります。公開情報によると、2022 年にリリースされた iOS16 は非常に安定していてパフォーマンスの高いバージョンであると考えられており、ユーザーはその全体的なエクスペリエンスに非常に満足しています。また、iOS16では新機能の追加や日常の使用感の向上も多くのユーザーからご好評をいただいております。特に最新のバッテリー寿命、信号性能、加熱制御に関して、ユーザーからのフィードバックは比較的好評です。ただし、iPhone14を考慮すると、

プログラミング言語 PHP は、さまざまなプログラミング ロジックやアルゴリズムをサポートできる、Web 開発用の強力なツールです。その中でも、フィボナッチ数列の実装は、一般的で古典的なプログラミングの問題です。この記事では、PHP プログラミング言語を使用してフィボナッチ数列を実装する方法を、具体的なコード例を添付して紹介します。フィボナッチ数列は、次のように定義される数学的数列です。数列の最初と 2 番目の要素は 1 で、3 番目の要素以降、各要素の値は前の 2 つの要素の合計に等しくなります。シーケンスの最初のいくつかの要素

4月11日、ファーウェイはHarmonyOS 4.2 100台のアップグレード計画を初めて正式に発表し、今回は携帯電話、タブレット、時計、ヘッドフォン、スマートスクリーンなどのデバイスを含む180台以上のデバイスがアップグレードに参加する予定だ。先月、HarmonyOS4.2 100台アップグレード計画の着実な進捗に伴い、Huawei Pocket2、Huawei MateX5シリーズ、nova12シリーズ、Huawei Puraシリーズなどの多くの人気モデルもアップグレードと適応を開始しました。 HarmonyOS によってもたらされる共通の、そして多くの場合新しい体験を楽しむことができる Huawei モデルのユーザーが増えることになります。ユーザーのフィードバックから判断すると、HarmonyOS4.2にアップグレードした後、Huawei Mate60シリーズモデルのエクスペリエンスがあらゆる面で向上しました。特にファーウェイM

Huawei 携帯電話に WeChat クローン機能を実装する方法 ソーシャル ソフトウェアの人気と人々のプライバシーとセキュリティの重視に伴い、WeChat クローン機能は徐々に人々の注目を集めるようになりました。 WeChat クローン機能を使用すると、ユーザーは同じ携帯電話で複数の WeChat アカウントに同時にログインできるため、管理と使用が容易になります。 Huawei携帯電話にWeChatクローン機能を実装するのは難しくなく、次の手順に従うだけです。ステップ 1: 携帯電話システムのバージョンと WeChat のバージョンが要件を満たしていることを確認する まず、Huawei 携帯電話システムのバージョンと WeChat アプリが最新バージョンに更新されていることを確認します。
