如何用谷物序列化 boost::uuid

How to serialize a boost::uuid with cereal

本文关键字:uuid boost 序列化 何用谷      更新时间:2023-10-16

尝试序列化这个简单的类:

class Data
{
public:
    Data();
    Data(boost::uuids::uuid id);
    Data(const Data&) = delete;
    Data& operator=(const Data&) = delete;
    inline boost::uuids::uuid getGuid() { return guid; }
    template <class Archive>
    void serialize(Archive & ar)
    {
        ar(guid);
    }
private:
    boost::uuids::uuid guid;
};

但是我收到此错误消息

error C2338: Trying to serialize an unserializable type with an output archive. 

Poiting to uuid。启用此功能的提升序列化方法是添加

#include <boost/uuid/uuid_serialize.hpp>

但这不适用于开箱即用的谷物。谷物文件说

谷物档案在 std::ostream 或 std::istream 对象上运行。

所以我尝试在定义但没有运气的地方添加标题

#include <boost/uuid/uuid_io.hpp>

这适用于谷物 JSON 存档。我还在评论中包含了为二进制存档执行此操作的方法。

#ifndef CEREAL_TYPES_BOOST_UUID_
#define CEREAL_TYPES_BOOST_UUID_
#include <boost/uuid/uuid.hpp>
#include <boost/uuid/uuid_io.hpp>
#include <boost/lexical_cast.hpp>
namespace cereal
{
    template <class Archive> inline
        void save(Archive& ar, boost::uuids::uuid const& uuid)
    {
        std::string val = boost::lexical_cast<std::string>(uuid);
        ar(val);
        // Other approach, probably better for binary
        //ar(make_size_tag(static_cast<size_type>(uuid.size())));
        //for (auto it = uuid.begin(), end = uuid.end(); it != end; ++it)
        //  ar(*it);
    }
    template <class Archive> inline
        void load(Archive& ar, boost::uuids::uuid& uuid)
    {
        std::string val;
        ar(val);
        uuid = boost::lexical_cast<boost::uuids::uuid>(val);
        // Other approach, probably better for binary
        //size_type size;
        //ar(make_size_tag(size));
        //for (auto it = uuid.begin(), end = uuid.end(); it != end; ++it) {
        //  uint8_t val;
        //  ar(val);
        //  *it = val;
        //}
    }
} // namespace cereal
#endif // CEREAL_TYPES_BOOST_UUID_

事实上

谷物档案在 std::ostream 或 std::istream 对象上运行。

(完全)并不意味着它使用 IO 流运算符(>>、<<)。这只是存档实现。

您必须实现免费函数serialize让 Cereal 了解您的类型。您应该能够重用 uuid_serialize.hpp 中所示的实现。很有可能你应该简单地

  • 将这些定义移动到 Cereal 命名空间(最好)或 boost::uuids 命名空间(将来可能会发生冲突)中,以便 ADL 找到它们
  • 最好将 UUID 视为简单的字节数组(AFAIR boost::uuids::uuid 是一种 POD 数据类型)