protobuf是Google开发的一个开源序列化框架,类似XML,JSON,其最大的特点是基于二进制,比传统的XML表示同样一段内容要短小得多。通过protobuf,可以很轻松的调用相关方法来完成业务数据的序列化与反序列化。但是,protobuf在使用当中也发现一些问题,数据在protobuf序列化后无法反序列化,这就与protobuf的目的违背。下面我们来看下这个问题吧。
发送过来的字节串
下面是我收到数据后解析出来的
printf("msg from MEC recved len: %d\n",payload_len);
printf("----------udm share1-----------------\n");
for(int i = 0;i < payload_len;i++)
{
printf("%d ",payloadptr[i]);
}
printf("\n");
这两段数据负数与下面接收到的正数绝对值相加为256,也就是正数与负数互为补码。
所以消息收发,是没有问题的,但是解析出来的数小数部分全是0
首先定义一个protobuf结构,如下:
message Person {
required int32 mid = 1;
required int32 type = 2;
}
再来看下这段代码,看似没有问题:
Person psn0;
psn0.set_mid(1);
psn0.set_type(0);
string str0;
psn0.SerializeToString(&str0);
// 由于网络传输, string 需要转换成 char*
const char* buf = str0.c_str();
// 如果是这样解析,将会引起bug
string str1 = buf;
People psn1;
bool ret1 = psn1.ParseFromString(str1);
if (ret1)
{
cout << "str1 parse success" << endl;
}
else
{
cout << "str1 parse fail" << endl;
}
// 这样解析不会引起bug
string str2;
str2.assign(buf, str0.length());
People psn2;
bool ret2 = psn2.ParseFromString(str2);
if (ret2)
{
cout << "str2 parse success" << endl;
}
else
{
cout << "str2 parse fail" << endl;
}
运行结果,如下:
str1 parse fail
str2 parse success
这样就可以解析到正确的数据了。
这里解释一下原因:在以上例子中,protobuf 序列化的 string 数据里含有 ‘\0‘ 字符。如果以 char* 取protobuf序列化后的数据,将会丢失部分数据。
虽然这个严格意义来讲也不算protobuf的bug,但在protobuf的使用中一般很少人注意protobuf 的数据中带有 ‘\0‘ 字符,这就导致了bug的发生。
//将消息转换成protobuf的字节流
std::string str = msg_vin_up_.SerializeAsString();
if(str.length() > 0)
{
len = str.length();
char *msg_data = (char*)str.c_str(); //string类型转换成char*
}
参考:
http://blog.csdn.net/mycwq/article/details/19619875
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)