C++ Cereal序列化库的使用

C++ Cereal 库使用指南

Cereal 是一个轻量级的 C++ 序列化库,用于将对象序列化为二进制、XML 或 JSON 格式,以及从这些格式反序列化。它支持标准库类型和用户自定义类型的序列化,且无需修改原有类定义。

基本用法

1. 安装与包含

复制代码
#include <cereal/archives/binary.hpp>
#include <cereal/types/vector.hpp>  // 包含对vector的支持
#include <cereal/types/string.hpp> // 包含对string的支持

2. 基本序列化/反序列化

复制代码
#include <fstream>
#include <iostream>
#include <vector>

struct MyData {
    int x;
    double y;
    std::string z;
    std::vector<int> nums;

    // 序列化函数模板
    template<class Archive>
    void serialize(Archive & archive) {
        archive(x, y, z, nums); // 序列化所有成员
    }
};

int main() {
    MyData data1{42, 3.14, "hello", {1, 2, 3}};
    
    // 序列化到文件
    {
        std::ofstream os("data.bin", std::ios::binary);
        cereal::BinaryOutputArchive archive(os);
        archive(data1); // 写入数据
    }

    // 从文件反序列化
    MyData data2;
    {
        std::ifstream is("data.bin", std::ios::binary);
        cereal::BinaryInputArchive archive(is);
        archive(data2); // 读取数据
    }

    std::cout << data2.x << ", " << data2.y << ", " << data2.z << std::endl;
    return 0;
}

高级特性

1. 分离加载和保存

复制代码
struct MyClass {
    int a;
    std::string b;

    template<class Archive>
    void save(Archive & archive) const {
        archive(a, b);
    }

    template<class Archive>
    void load(Archive & archive) {
        archive(a, b);
    }
};

2. 版本控制

复制代码
struct VersionedData {
    int x;
    std::string y;

    template<class Archive>
    void serialize(Archive & ar, const std::uint32_t version) {
        ar(x, y);
        if(version >= 2) {
            // 版本2新增的字段
        }
    }
};

// 注册版本
CEREAL_CLASS_VERSION(VersionedData, 2)

3. 指针和智能指针

复制代码
struct Node {
    int value;
    std::shared_ptr<Node> next;

    template<class Archive>
    void serialize(Archive & ar) {
        ar(value, next);
    }
};

4. 继承

复制代码
struct Base {
    int x;
    template<class Archive>
    void serialize(Archive & ar) {
        ar(x);
    }
};

struct Derived : Base {
    double y;
    template<class Archive>
    void serialize(Archive & ar) {
        ar(cereal::base_class<Base>(this), y);
    }
};

不同格式支持

1. 二进制格式

复制代码
#include <cereal/archives/binary.hpp>

// 输出
std::ofstream os("data.bin", std::ios::binary);
cereal::BinaryOutputArchive archive(os);
archive(data);

// 输入
std::ifstream is("data.bin", std::ios::binary);
cereal::BinaryInputArchive archive(is);
archive(data);

2. JSON格式

复制代码
#include <cereal/archives/json.hpp>

// 输出
std::ofstream os("data.json");
cereal::JSONOutputArchive archive(os);
archive(cereal::make_nvp("MyData", data)); // 使用命名

// 输入
std::ifstream is("data.json");
cereal::JSONInputArchive archive(is);
archive(data);

3. XML格式

复制代码
#include <cereal/archives/xml.hpp>

// 输出
std::ofstream os("data.xml");
cereal::XMLOutputArchive archive(os);
archive(cereal::make_nvp("MyData", data));

// 输入
std::ifstream is("data.xml");
cereal::XMLInputArchive archive(is);
archive(data);

实际应用示例

1. 序列化自定义类

复制代码
class Person {
public:
    Person() = default;
    Person(std::string n, int a) : name(n), age(a) {}
    
private:
    std::string name;
    int age;
    std::vector<std::string> friends;
    
    friend class cereal::access;
    
    template<class Archive>
    void serialize(Archive & ar) {
        ar(name, age, friends);
    }
};

2. 序列化STL容器

复制代码
std::map<int, std::string> myMap = {{1, "one"}, {2, "two"}};
std::vector<double> myVec = {1.1, 2.2, 3.3};

{
    std::ofstream os("container.bin", std::ios::binary);
    cereal::BinaryOutputArchive archive(os);
    archive(myMap, myVec);
}

3. 多态类型序列化

复制代码
struct Base {
    virtual void foo() = 0;
    virtual ~Base() = default;
    
    template<class Archive>
    void serialize(Archive & ar) {}
};

struct Derived : Base {
    int x;
    void foo() override {}
    
    template<class Archive>
    void serialize(Archive & ar) {
        ar(cereal::base_class<Base>(this), x);
    }
};

// 注册多态类型
CEREAL_REGISTER_TYPE(Derived)

注意事项

  1. 默认构造函数:反序列化的类必须有可访问的默认构造函数

  2. 内存管理:序列化原始指针时需要特别小心,推荐使用智能指针

  3. 版本兼容性:修改类结构后,旧版本数据可能无法正确反序列化

  4. 性能:二进制格式通常比文本格式更高效

  5. 可移植性:二进制数据在不同平台间可能存在兼容性问题

Cereal 是一个强大而灵活的序列化库,通过简单的接口实现了复杂对象的序列化功能,非常适合需要持久化或网络传输C++对象的场景。

深入学习链接(参考🔗)C++序列化Cereal库的使用-CSDN博客

相关推荐
1白天的黑夜11 分钟前
递归-206.反转链表-力扣(LeetCode)
数据结构·c++·leetcode·链表·递归
Fcy6489 分钟前
C++ vector容器的解析和使用
开发语言·c++·vector
无限进步_19 分钟前
C语言文件操作全面解析:从基础概念到高级应用
c语言·开发语言·c++·后端·visual studio
_OP_CHEN22 分钟前
C++基础:(十五)queue的深度解析和模拟实现
开发语言·c++·stl·bfs·queue·容器适配器·queue模拟实现
sulikey22 分钟前
一文彻底理解:如何判断单链表是否成环(含原理推导与环入口推算)
c++·算法·leetcode·链表·floyd·快慢指针·floyd判圈算法
起床气23335 分钟前
C++海战棋开发日记(序)
开发语言·c++
APItesterCris1 小时前
TypeScript 与淘宝 API:构建类型安全的商品数据查询前端 / Node.js 服务
开发语言·php
ftpeak1 小时前
《Cargo 参考手册》第二十一章:Cargo 包命令
开发语言·rust
_码力全开_1 小时前
P1005 [NOIP 2007 提高组] 矩阵取数游戏
java·c语言·c++·python·算法·矩阵·go