MessagePack,c++:如何在c++11枚举类中使用MSGPACK_DEFINE

4
下面的示例无法编译,报错为:
In file included from /usr/include/msgpack.hpp:18:
/usr/include/msgpack/object.hpp:211:3: error: member reference base type 'logd::log_level' is not a structure or union

另一个枚举类对应的错误。我的问题是,如何使用msgpack的c++ api序列化具有C++11 enum class类型成员的类?

#ifndef LOG_MSG_HPP_
#define LOG_MSG_HPP_

#include <stdlib.h>
#include <msgpack.hpp>

/** @namespace logd */
namespace logd {

enum class log_level { SILENT,... DEBUG };

enum class log_domain { AI, ...  MISC };

class log_msg {
    public:
        log_msg(log_level lev, log_domain dom, std::string msg);
        log_level level();
        log_domain domain();
        std::string message();
        ~log_msg();
        MSGPACK_DEFINE(lev_, dom_, msg_);

    private:
        log_msg();
        log_level   lev_ {log_level::DEBUG};
        log_domain  dom_ {log_domain::MISC};
        std::string msg_ {"No message given."};
};
} /* namespace logd */
#endif /* LOG_MSG_HPP_ */

注意: 由于这些枚举是我的,所以我可以愉快地修改它们以使msgpack更容易处理。不幸的是,我在他们的文档或谷歌的前几页上找不到任何有关此主题的参考资料。由于我对c++相当新手,所以无法从阅读他们的头文件/源代码中确定如何处理。

3个回答

6
你可以使用MSGPACK_ADD_ENUM()宏。该宏从版本1.0.0开始受支持,我建议使用版本1.1.0或更高版本。
参考: https://github.com/msgpack/msgpack-c/wiki/v1_1_cpp_adaptor#enums 我已将MSGPACK_ADD_ENUM()应用到你的代码中。
#ifndef LOG_MSG_HPP_
#define LOG_MSG_HPP_

#include <stdlib.h>
#include <msgpack.hpp>

/** @namespace logd */
namespace logd {

enum class log_level { SILENT,DEBUG };

enum class log_domain { AI, MISC };

class log_msg {
    public:
        log_msg(log_level lev, log_domain dom, std::string msg):lev_(lev), dom_(dom), msg_(msg) {}
        log_level level() { return lev_;}
        log_domain domain() { return dom_;}
        std::string message() { return msg_; }
        ~log_msg() {}
        MSGPACK_DEFINE(lev_, dom_, msg_);
        log_msg() = default;

    private:
        log_level   lev_ {log_level::DEBUG};
        log_domain  dom_ {log_domain::MISC};
        std::string msg_ {"No message given."};
};
} /* namespace logd */

MSGPACK_ADD_ENUM(logd::log_level);
MSGPACK_ADD_ENUM(logd::log_domain);


#endif /* LOG_MSG_HPP_ */

#include <sstream>
#include <cassert>

int main() {
    logd::log_msg lm { logd::log_level::SILENT, logd::log_domain::AI, "hello" };
    std::stringstream ss;
    msgpack::pack(ss, lm);

    msgpack::object obj = msgpack::unpack(ss.str().data(), ss.str().size()).get();
    logd::log_msg lm2 = obj.as<logd::log_msg>();
    assert(lm.level() == lm2.level());
    assert(lm.domain() == lm2.domain());
    assert(lm.message() == lm2.message());
}

0

我找到了另一种解决方案,可能更加优雅一些(好吧,它并不是那么完美):

MSGPACK_DEFINE((int&)lev_, (int&)dom_, msg)

这样你就可以保留原始的枚举而不需要创建新的联合体。

反序列化方法应该可以无问题地工作。


0
似乎可行的一种方法是将枚举封装在联合体中...
union log_level_t {
    log_level lev;
    int       levint;
}

...

    log_level_t lev_;

...

MSGPACK_DEFINE(lev_.levint, dom_.domint, msg);

这似乎是一种糟糕的方法,但它能够工作。


这是使用MessagePack枚举类的唯一方法吗?不太优雅! - Kill KRT

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接