gpt4 book ai didi

c++ - 存储一组序列化的 protobuf 对象

转载 作者:塔克拉玛干 更新时间:2023-11-03 01:51:25 26 4
gpt4 key购买 nike

我在 Protobuf 对象上使用 SerializeToString,然后将字符串存储在数据库中。

但是,有时我有一组这样的对象。我想存储整个序列化数组,为此我需要在序列化字符串之间添加一些分隔符字符串。

根据documentation我已经看到,该字符串只是一个字节数组,因此我没有得到关于其内容的任何 promise 。

这里最好的方法是什么?

我不知道数组的长度,因为对象可能会在我们进行时附加到它,我希望它在整个过程中都存储在数据库中。

最佳答案

假设,您的 protobuf 消息 如下所示:

message Object
{
... = 1;
... = 2;
... = 3;
}

然后在同一个文件中再引入 1 个 message,它是这些 Object 的集合。

message Objects
{
repeated Object array = 1;
}

因此,当您有很多元素时,您可以简单地使用 Objects 并在 Objects 本身上使用 SerializeAsString()。这将节省您单独序列化单个 Object 并放置您自己的手工定界符的工作。您可以使用 Objects 的单个实例序列化所有 Object
通过这种方法,您可以将所有解析和序列化工作也委托(delegate)给 Protobuf 库。我在我的项目中使用了它,它非常有用。

此外,明智地使用Objects 还可以避免制作额外的Object 拷贝。您可以向其中添加项目并使用索引进行访问。 protobuf 的 repeated 字段符合 C++11,因此您也可以将它与迭代器或增强的 for 循环一起使用。


重要的是要注意,当您将 Objects::SerializeAsString() 的输出存储到文件中时,您应该首先输入该 string 的长度,然后通过实际的序列化字符串。读取时,可以先读取长度,再读取总字节数。为了便于使用,我扩展了 std::fstream 并重载了以下方法:

struct fstreamEncoded : std::fstream
{
// other methods
void // returns `void` to avoid multiple `<<` in a single line
operator<< (const string& content)
{ // below casting would avoid recursive calling of this method
// adding `length() + 1` to include the protobuf's last character as well
static_cast<std::fstream&>(*this) << (content.length() + 1) << "\n" << content << std::endl;
}

string
getline ()
{
char length_[20] = {};
std::istream::getline(length_, sizeof(length_) - 1);
if(*length_ == 0)
return "";

const size_t length = std::atol(length_); // length of encoded input
string content(length, 0); // resize the `string`
read(&content[0], length); // member of `class istream`
return content;
}
}

以上只是一个例子。您可以根据您的项目需要进行操作。

关于c++ - 存储一组序列化的 protobuf 对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38321008/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com