0

0

c++中如何使用FlatBuffers进行高效序列化? (零拷贝原理)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-10 05:35:23

|

786人浏览过

|

来源于php中文网

原创

FlatBuffers 零拷贝核心是直接内存布局+offset访问,不解析不复制不分配;需用flatc生成头文件并包含flatbuffers/flatbuffers.h;Builder应栈上构造并用Release()转移所有权;读取时须空指针检查且不可越界访问。

c++中如何使用flatbuffers进行高效序列化? (零拷贝原理)

FlatBuffers 的零拷贝原理到底怎么工作的?

FlatBuffers 不做传统序列化(比如把对象转成 JSON 字符串再解析),而是把数据直接按内存布局写进一块连续 buffer 里,结构体字段用 offset 定位,读取时跳过解析步骤,直接指针偏移访问——这才是零拷贝的核心:不复制、不解析、不分配临时对象。

关键前提是:flatc 编译器生成的 C++ 代码是只读访问器(TableVector 等),所有字段访问都是计算 offset + reinterpret_cast,没有 new、没有 memcpy、没有 string 构造。

  • buffer 必须保持 alive(不能局部 std::vector 返回后析构)
  • 必须用 GetRoot() 获取根对象指针,不能直接 reinterpret_cast
  • 所有字段访问都依赖 buffer 对齐(默认 16 字节对齐,可通过 --align 调整)

如何用 flatc 生成 C++ 访问器并正确链接?

先写 schema(monster.fbs),再用 flatc 生成头文件;C++ 项目不需要链接额外库,但必须包含 FlatBuffers 运行时头文件(flatbuffers/flatbuffers.h)和生成的 *.h

常见错误:只加了生成头文件,忘了 #include "flatbuffers/flatbuffers.h",编译报 ‘flatbuffers::Table’ has not been declared

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

  • flatc --cpp monster.fbs 生成 monster_generated.h
  • CMake 中确保 include path 包含 FlatBuffers 源码目录(或安装路径下的 include
  • 无需链接 -lflatbuffers —— FlatBuffers C++ 是 header-only(除可选的 flatbuffers/util.h 工具函数外)

构建 FlatBuffer 时为什么不能直接 new 或 stack 分配 Builder?

flatbuffers::FlatBufferBuilder 内部持有 std::vector,会动态扩容。如果在上声明(如 FlatBufferBuilder builder;),没问题;但如果用 new FlatBufferBuilder,后续调用 Finish() 返回的指针可能失效——因为 builder.Release() 会移交 vector 内存所有权,而 new 出来的对象生命周期难控。

Ink For All
Ink For All

AI写作和营销助手,精心设计的 UI

下载

更危险的是:把 builder.GetBufferPointer() 存下来,然后让 builder 析构,指针立刻悬空。

  • 推荐方式:栈上构造,Finish() 后立即用 builder.Release() 拿到 std::vector 所有权
  • 若需长期持有 buffer,用 std::move(builder.Release()) 转移数据,不要保留 builder 实例
  • 避免 uint8_t* ptr = builder.GetBufferPointer(); ... return ptr; —— 这是典型悬垂指针
flatbuffers::FlatBufferBuilder builder(1024);
auto name = builder.CreateString("Orc");
auto monster = CreateMonster(builder, name);
builder.Finish(monster);

// ✅ 正确:转移所有权 auto buffer = std::move(builder.Release());

// ❌ 错误:builder 析构后 buffer.data() 失效 const uint8_t* ptr = buffer.data(); // ... 之后 ptr 仍有效,因为 buffer 还活着

读取时如何安全访问嵌套 table 和 vector?

FlatBuffers 的 TableVector 都是逻辑视图,底层共享同一块 buffer。访问 monster->inventory() 返回的是 const Vector* ,不是拷贝数据;访问 monster->name()->str() 返回的是 const char*,指向 buffer 内原始字节。

陷阱在于:string 字段为空时,monster->name() 返回 nullptr,不是空字符串;vector 为空时,vec->size() 为 0,但 vec->data() 仍合法(只是不可读)。

  • 永远检查 monster->name() 是否非空,再调用 ->str()
  • vector 访问前用 vec && vec->size() > 0 判断,避免 vec->Get(i) 越界(不抛异常,行为未定义)
  • 不要对返回的 const char*strlen —— FlatBuffers string 是带 length prefix 的,应使用 str()->str()(已封装)或 str()->length()
auto monster = GetMonster(buffer.data());
if (monster && monster->name()) {
  std::cout << monster->name()->str() << "\n"; // ✅ 自动处理 null-terminator 和 length
}
if (auto inv = monster->inventory()) {
  for (size_t i = 0; i < inv->size(); ++i) {
    std::cout << static_cast(inv->Get(i)) << " ";
  }
}

零拷贝不是魔法,它把内存管理责任完全交给了你:buffer 生命周期、对齐保证、空值判断、越界防护,每一步漏掉都会导致静默崩溃或数据错乱。最常被忽略的是 buffer 的持有者——它必须比所有访问它的 Table* 活得更久。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

408

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

532

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

309

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

74

2025.09.10

string转int
string转int

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

315

2023.08.02

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

520

2023.09.20

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

253

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

206

2023.09.04

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

15

2026.01.09

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.2万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.1万人学习

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

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