【1】cpp_redis (Windows C++ Redis客户端)(C++11实现)官方最新源码编译
正文如下
reply
reply是cpp_redis访问redis服务返回的唯一类型,该类型可以包含各种情况下的服务端数据。
reply的结构
namespace cpp_redis {
class reply {
private:
type m_type;
std::vector<cpp_redis::reply> m_rows;
std::string m_strval;
int64_t m_intval;
};
普通情况下单个值放到m_strval或者m_intval中
之所以能够包含各种类型的返回结果,是因为该结构为递归结构m_rows为vector<reply>;最终都会以一个string或者int存放
取得reply中存放的值
reply中有点类似C++中的union类型,或者Rapidjson中的Value类型,也就是符合类型。所以当你取值的时候最好要判断一下,如果取错了类型会抛异常。
当然,自己放进去的值如果自己清楚,也不用判断,直接取值。直接取值的话用下面的函数从reply中得到对应的值:
const std::vector<reply>& as_array(void) const;
const std::string& as_string(void) const;
int64_t as_integer(void) const;
1 as_string() 得到reply中保存的m_strval值
2 as_integer() 得到reply中保存的m_intval值
3 as_array() 得到reply中保存的vector<reply>;再对vector中的每个reply调用as_string取得m_strval字符串,或者对vector中的每个reply调用as_integer取得m_intval整数;如果多层结构,就继续增加内循环取出数据。
4 如果一个key在redis中不存在,请求返回的值类型为空值。
bool
reply::is_null(void) const {
return m_type == type::null;
}
5 返回多个值的时候某些key如果在redis中不存在:
如果请求返回的是vector<reply>那一定是因为你请求的时候发送的是#mget("name", vector<string> many_keys).此时返回的vector<reply>中,某些reply可能因为你发送的key在redis中并不存在而为上面的空值。
6 从空值中取数据之前要先判断才能取(避免不必要的异常发生)
bool
reply::is_array(void) const {
return m_type == type::array;
}
bool
reply::is_string(void) const {
return is_simple_string() || is_bulk_string() || is_error();
}
bool
reply::is_simple_string(void) const {
return m_type == type::simple_string;
}
bool
reply::is_bulk_string(void) const {
return m_type == type::bulk_string;
}
bool
reply::is_error(void) const {
return m_type == type::error;
}
bool
reply::is_integer(void) const {
return m_type == type::integer;
}
下面的例子以往redis中的hash容器存取数据为例,完整的展示了上面提到的内容:
#include <cpp_redis/cpp_redis>
#include <iostream>
#include <utility>
#include "TestRedis.h"
using namespace std;
#ifdef _WIN32
#include <Winsock2.h>
#endif /* _WIN32 */
#pragma comment( lib, "ws2_32.lib")//最好的方法是包含在项目属性中,因为那样可以根据Debug、Release、x86、x64来区分。这里仅仅是为了突出引用了这个库写在这里
bool StartWSA(void)
{
//! Windows netword DLL init
WORD version = MAKEWORD(2, 2);
WSADATA data;
if (WSAStartup(version, &data) != 0)
{
std::cerr << "WSAStartup() failure" << std::endl;
return false;
}
return true;
}
void StopWSA(void)
{
WSACleanup();
}
int main(void)
{
//启动Windows网络通信库
if (!StartWSA())
{
return -1;
}
try
{
//! Enable logging
cpp_redis::active_logger = std::unique_ptr<cpp_redis::logger>(new cpp_redis::logger);
cpp_redis::redis_client client;
client.connect("127.0.0.1", 6379, [](cpp_redis::redis_client&)
{
std::cout << "client disconnected (disconnection handler)" << std::endl;
});
vector<pair<string, string>> vecPair = { {"key1", "value1"},{ "key2", "value2" }, {"key3", "value3"} };
client.hmset("hash", vecPair);
vector<string> vecKey = { "key1", "key2", "key4" };
client.hmget("hash", vecKey, [](cpp_redis::reply& _reply)
{
for each (auto& var in _reply.as_array())
{
if (var.is_string())
{
cout << var.as_string() << endl;
}
}
});
// synchronous commit, no timeout
client.sync_commit();
}
catch (cpp_redis::redis_error e)
{
std::cout << e.what() << endl;
}
//关闭Windows网络通信库
StopWSA();
return 0;
}
运行结果:
我们创建了3个key,取的时候故意取了一个不存在的key4,所以返回的结果是nill。这时候我们对返回的vector<reply>每个元素都做了is_string()判断,如果不判断就会抛出异常,这个时候使用异常就没必要了(代码结构不清晰)。