Reputation: 444
At first I have this simple protobuf file
message messagetest
{
...
repeated float samples = 6;
....
}
Which creates a headerfile with this methods
//repeated float samples = 6;
inline int samples_size() const;
inline void clear_samples();
static const int kSamplesFieldNumber = 6;
inline float samples(int index) const;
inline void set_samples(int index, float value);
inline void add_samples(float value);
inline const ::google::protobuf::RepeatedField< float >& samples() const;
inline ::google::protobuf::RepeatedField< float >* mutable_samples();
What I'm basically doing is to copy all data one by one in a for loop.
int main(int argc, char** argv)
{
messagetest fMessage;
vector<float> fData (1000, 0);
// Create 1000 random values
for (int i = 0; i < fData.size(); i++)
{
fData[i] = rand() % 1001;
}
for (int j = 0; j < fData.size(); j++)
{
fMessage.add_samples(fData[j]);
}
return 0;
}
But I want to use a method like memcpy to accelerate the copy process. It is just an idea that comes to my mind. If it's completely wrong correct me. The last declaration in the headerfile is:
inline ::google::protobuf::RepeatedField< float >* mutable_samples();
I have no idea what this method does (lack of skill). But it kind of looks like a vector. Maybe that's the solution for my problem. If so, I have no idea how to implement it.
Upvotes: 27
Views: 60488
Reputation: 3235
As an alternative to the excellent answer posted by @mgild, Assign
can also be used in this situation, so the data would not be copied to a temporary before being moved.
fMessage.mutable_samples()->Assign(fData.begin(), fData.end())
Credit to Marek R's answer
Upvotes: 2
Reputation: 7253
@mgild's answer will implicitly call the RepeatedField(Iter begin, Iter end)
constructor to create a temporary to be move assigned. Same with @nazgul's answer, which explicitly creates a temporary RepeatedField and swaps it.
Much simpler, avoiding the creation of a new object would be:
fMessage.mutable_samples()->Add(fData.begin(), fData.end())
If samples
field is not empty already, you can call Clear
method prior.
Internally, this uses std::copy
(so long as fData
is a forward iterator), so will be just as fast as any memcpy implementation you come up with.
Upvotes: 2
Reputation: 519
I found the shortest way to copy vector into repeated field as this:
google::protobuf::RepeatedField<float> data(fData.begin(), fData.end());
fMessage.mutable_samples()->Swap(&data);
It is probably also faster than yours since it avoids initial iteration and setting values to 0.
Upvotes: 23
Reputation: 43
fMessage.mutable_samples()
return an array of pointer of samples : [*sample1, *sample2, sample3, ...].
&fData[0]
is the address of first element of fData.
memcpy(fMessage.mutable_samples()->mutable_data(),
&fData[0],
sizeof(float)*fData.size());
So I do not think the code above can successfully fill data from fData to fMessage. It's totally wrong!
Upvotes: -3
Reputation: 794
Since this isn't here yet and I like one-liners:
*fMessage.mutable_samples() = {fData.begin(), fData.end()};
Upvotes: 57