Protobuf 调试踩坑记录

这是一次记录在使用protobuf中踩坑的记录,做一次记录整理,也希望能给其他遇到坑的同学一些帮助。

目录

背景:

踩坑记录

problem 1:

solution:

problem 2:

solution:

problem 3:

 solution:

转存失败重新上传取消​编辑problem 4:

 solution:

总结


背景:

本次开发环境是一个跨平台的通信模板,其中一端平台不支持proto生成的c++的文件,只能使用nanopb生成c的文件,而另一端用的c++的,跨平台通信涉及数据内存拷贝,编码解码工作

ps:关于protobuf和nanopb使用和介绍一大堆教程文档就不做赘述了

踩坑记录

problem 1:

结构体赋值传输过去后直接进行数据拷贝,结果报错了,定位问题:

写proto时候会写一个message,如果是一个结构体

nanopb生成的.h头文件中也会生成一个结构体,使用nanopb时候可以直接定义一个结构体变量,然后进行赋值,

使用c++会生成一个hpp文件,它会定义一个结构体同名的类

在调试中添加打印查看结构体数据长度:

使用nanopb端打印

 c++打印:

 结果:

 (此处坑看着很简单,常识性错误,提醒使用第三方工具一定要看它生成文件)

solution:

它们的内存和长度完全不同,不可以将结构体数据直接进行拷贝,类似“memcpy"是一定错误的,必须将结构进行编码传输过去后再进行解码工作才可以。

problem 2:

在上面问题解决添加nanopb编码工作,结果编码时候后也没提示报错程序运行直接崩溃

调试过程耽搁很久,但最后问题也很简单

在proto定义中

repeated:在格式正确的消息中,此字段类型可以重复零次或多次。系统会保留重复值的。

这个消息类型生成的类成员在c++中类似容器,可以动态分配内存大小

然而在nanopb中,repeated关键字必须指定最大长度,让c语言一次分配好内存大小

solution:

在proto定义路径下:再写一个与proto同名的.options文件

如下:

Protobuf 调试踩坑记录_第1张图片

 这样再使用nanopb生成消息文件,在生成头文件中多了消息的最大长度

 这时候就正常编码了

problem 3:

3.第二步添加编码后,依然无法正确发送数据,添加打印定位到编码失败报错-array max size exceeded

Protobuf 调试踩坑记录_第2张图片

 

 solution:

修改结构体长度,之前定义的结构体长度算都有10k了,并将data_tmp大小改为生成的头文件中定义的结构大小

problem 4:

在nanopb编码后并且程序成功运行,但是当c++解码后probobuf内容都是零;

Protobuf 调试踩坑记录_第3张图片

 上面截图可以看到我先把结构体数据打印出来,再进行编码,编码后直接解码打印查看:

Protobuf 调试踩坑记录_第4张图片

结构体有值,编码后再解码发现此时就已经为零。

最终调试半天发现,在nanopb中,

所有repeated消息在生成的结构体会有个成员pb_size_t count这个值必须指定为实际使用的值,

类似one of 的消息,也必须指定使用哪一个,这样才能成功编码

当给这些变量赋值后:再看打印,成功解析Protobuf 调试踩坑记录_第5张图片

 solution:

编码后一定注意生成的结构体内容,像动态消息  有pb_size_t的变量一定按实际填充数量给它赋值

总结

上面这些看着都是一些小问题,但在实际开发调试中耽搁好几天时间,重复的代码修改很枯燥,非常耗费精力和时间, 所以整理下来,在开发中,要细心有耐心,保持质疑,不要相信玄学,尤其对第三方库,一定多去看使用文档,多查多问,保持记录的习惯。

你可能感兴趣的:(protobuf,protobuf,nanopb)