最近遇到一个问题,需要将多个 protobuf 序列化后存为文件,于是定了如下的协议格式。规定 PB1_size 以 4 字节大小存储在文件中。
[PB1_size][PB1_body][PB2_size][PB2_body]...[]
PBn_size 调用 fstream 流式操作以下列方式 write 写入
# include# include // msg 为 protobuf 结构体fstream fs;while (pb 未写完){ uint32_t pb1_size = msg.ByteSize(); fs.write(char*(&pb1_size), 4); msg.SerializeToOstream(&fs);}
char*(&pb1_size) 取得 uint32_t 数据最低位字节的地址并强制转换为 char*,这样可以将 pb1_size 的四个字节以从低到高的方式写入文件。
读 pb 大小代码如下:
fstream fs;char buffer[4];fs.read(buffer, 4);int size = atoi(buffer);
这样将 buffer 转为 size 会有问题,假设写入的 int 是 0x00000008,即 pb 大小为 8。写到文件里的二进制是 0x00000008。读到 buffer 里相当于字符串 "NUL NUL NUL BS",无法使用 atoi 来转换。
下面式例可以正常转换。
int main(){ char buffer_size[5]; buffer_size[0] = 56; buffer_size[1] = 48; buffer_size[2] = 48; buffer_size[3] = 48; cout << atoi(buffer_size) << endl;}