0

0

C++保存对象到二进制文件_序列化入门指南

穿越時空

穿越時空

发布时间:2026-01-31 12:25:02

|

424人浏览过

|

来源于php中文网

原创

C++标准库不提供通用序列化机制;reinterpret_cast(&obj)+write()仅对POD类型安全,含指针、虚函数或STL容器时会因堆内存未保存而崩溃;应手动实现serialize/deserialize或选用protobuf/msgpack等成熟库,并始终加入版本号管理。

c++保存对象到二进制文件_序列化入门指南

直接说结论:C++ 标准库不提供通用对象序列化机制,operator 和 fstream::write() 都不能安全地“一键保存任意对象”——尤其当对象含指针、虚函数、STL 容器或继承关系时。

为什么 reinterpret_cast(&obj) + write() 会出问题

这种写法只对 POD(Plain Old Data)类型有效,比如纯结构体且不含指针、引用、虚函数、非静态成员函数、std::string 等。一旦对象含 std::vectorstd::string,实际数据在堆上,write() 只存了上的指针值,读取时解引用必然崩溃或读到垃圾数据。

  • 常见错误现象:read()std::string::c_str() 段错误、vector.size() 返回极大随机数、反序列化后内容全乱
  • 虚表指针(vptr)是编译器插入的,跨平台/跨编译器/跨版本不可移植
  • 内存对齐和字节序(如 x86 小端 vs ARM 大端)会导致读写不一致

最简可行方案:手动定义 serialize()deserialize() 成员函数

适用于自定义类且结构稳定(如游戏存档、配置缓存),控制力强、无第三方依赖。

  • 把每个字段按确定顺序、确定大小逐个读写,绕过指针和容器内部布局
  • std::string:先写长度(uint32_t),再写字符数据;读时先读长度,再 resize()read()
  • std::vector:同理,先写 size(),再循环写每个 T(前提是 T 本身可序列化)
  • 避免直接操作 this 地址——用普通函数参数传递,更易测试和复用
struct Player {
    int level = 1;
    std::string name;
    std::vector inventory;

    void serialize(std::ostream& os) const {
        os.write(reinterpret_cast(&level), sizeof(level));
        uint32_t len = static_cast(name.length());
        os.write(reinterpret_cast(&len), sizeof(len));
        os.write(name.data(), len);
        len = static_cast(inventory.size());
        os.write(reinterpret_cast(&len), sizeof(len));
        for (int item : inventory) {
            os.write(reinterpret_cast(&item), sizeof(item));
        }
    }

    void deserialize(std::istream& is) {
        is.read(reinterpret_cast(&level), sizeof(level));
        uint32_t len;
        is.read(reinterpret_cast(&len), sizeof(len));
        name.resize(len);
        is.read(&name[0], len);
        is.read(reinterpret_cast(&len), sizeof(len));
        inventory.resize(len);
        for (int& item : inventory) {
            is.read(reinterpret_cast(&item), sizeof(item));
        }
    }
};

什么时候该换方案:别硬扛,用成熟库

当出现以下任一情况,手动序列化成本陡增,建议切换:

uBrand
uBrand

一站式AI品牌创建平台,在线品牌设计,AI品牌策划,智能品牌营销;uBrand帮助创业者轻松打造个性品牌!

下载

立即学习C++免费学习笔记(深入)”;

  • 类有继承体系(需处理基类字段、虚继承)
  • 需要跨语言交互(如和 Python/JS 通信)
  • 要求向后兼容(新增字段不影响旧版本读取)
  • 频繁修改结构,手动同步读写逻辑易出错

此时推荐:protobuf(需 .proto 描述)、msgpack(C++ 接口简洁)、或 Boost.Serialization(支持运行时类型信息,但二进制格式不跨平台)。它们都强制你声明“哪些字段参与序列化”,并自动处理容器、指针(含智能指针)、继承等边界情况。

最常被忽略的一点:二进制序列化不是“存完就完”,必须配套版本号管理——哪怕只是文件头加一个 uint32_t version = 1;。没有它,第一次改结构就等于废掉所有旧存档。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

483

2023.08.02

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

240

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

192

2025.07.04

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

483

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

545

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

113

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

200

2025.08.29

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1157

2023.10.19

2026赚钱平台入口大全
2026赚钱平台入口大全

2026年最新赚钱平台入口汇总,涵盖任务众包、内容创作、电商运营、技能变现等多类正规渠道,助你轻松开启副业增收之路。阅读专题下面的文章了解更多详细内容。

30

2026.01.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.4万人学习

Django 教程
Django 教程

共28课时 | 3.7万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号