protobuf删除字段后会变小吗

qq4398736 2015-08-16 11:39:01
直接用例子说话。
message CmdWrite
{
optional uint32 a = 1;
optional uint32 b = 2;
optional uint32 c = 3;
}

message CmdRead
{
optional uint32 a = 1;
optional uint32 c = 3;
}
其中CmdRead相当于CmdWrite删除了b字段。

CmdWrite cmdWrite;
cmdWrite.set_a(100);
cmdWrite.set_b(200);
cmdWrite.set_c(300);

std::string strWrite;
cmdWrite.SerializeToString(&strWrite);

CmdRead cmdRead;
bool bRet = cmdRead.ParseFromString(strWrite);
cmdRead.set_a(101);

std::string strRead;
cmdRead.SerializeToString(&strRead);
其中strWrite的size是8,strRead的size竟然也是8,按说CmdRead中少了b字段,序列化之后,strRead的size应该比strWrite小才对,为什么是一样的?
...全文
614 4 打赏 收藏 转发到动态 举报
写回复
用AI写文章
4 条回复
切换为时间正序
请发表友善的回复…
发表回复
赵4老师 2015-08-17
  • 打赏
  • 举报
回复
仅供参考:
void HexDump(char *buf,int len,int addr) {
    int i,j,k;
    char binstr[80];

    for (i=0;i<len;i++) {
        if (0==(i%16)) {
            sprintf(binstr,"%08x -",i+addr);
            sprintf(binstr,"%s %02x",binstr,(unsigned char)buf[i]);
        } else if (15==(i%16)) {
            sprintf(binstr,"%s %02x",binstr,(unsigned char)buf[i]);
            sprintf(binstr,"%s  ",binstr);
            for (j=i-15;j<=i;j++) {
                sprintf(binstr,"%s%c",binstr,('!'<buf[j]&&buf[j]<='~')?buf[j]:'.');
            }
            printf("%s\n",binstr);
        } else {
            sprintf(binstr,"%s %02x",binstr,(unsigned char)buf[i]);
        }
    }
    if (0!=(i%16)) {
        k=16-(i%16);
        for (j=0;j<k;j++) {
            sprintf(binstr,"%s   ",binstr);
        }
        sprintf(binstr,"%s  ",binstr);
        k=16-k;
        for (j=i-k;j<i;j++) {
            sprintf(binstr,"%s%c",binstr,('!'<buf[j]&&buf[j]<='~')?buf[j]:'.');
        }
        printf("%s\n",binstr);
    }
}
qq4398736 2015-08-16
  • 打赏
  • 举报
回复
看了一下protobuf源码,有这样一行: ::google::protobuf::UnknownFieldSet _unknown_fields_; protobuf会在反序列化时把不能识别的字段(比如CmdRead中被删掉的b字段)放在_unknown_fields_中,这样CmdRead在序列化时,会先序列化自己的字段a和c,接着判断_unknown_fields_字段是否为空,如果不为空(比如上面CmdRead中就存放了b字段的内容),接着会把_unknown_fields_字段的内容一起序列化。 这就解释了CmdWrite和CmdRead序列化后大小一样的问题,但是protobuf为什么这样做?为什么把未知(unknown)的字段也一起序列化?有没有宏或者开关,可以阻止未知字段的序列化。
QIUSQJF 2015-08-16
  • 打赏
  • 举报
回复
字节对齐……
qq4398736 2015-08-16
  • 打赏
  • 举报
回复
浏览了一下protobuf源码,问题解决了。 问题的原因上面说过了,解决办法不是调用一下cmdRead.DiscardUnknownFields(); 这个函数会把_unknown_fields_字段清空,这样cmdRead在序列化时,就不会把已经删掉的字段(b字段)序列化了。 这样序列化之后,cmdRead的大小就不是8了,是5

3,882

社区成员

发帖
与我相关
我的任务
社区描述
C/C++ 其它技术问题
社区管理员
  • 其它技术问题社区
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧