4

我正在使用 Nanopb,其中生成的 proto 文件中的字符串变量被转换为 pb_callback_t

所以,到目前为止,我正在尝试使用 nanopb 回调的测试示例;

bool encode_string(pb_ostream_t* stream, const pb_field_t* field, void* const* arg)
{
    char str[14] = "Hello world!";

    if (!pb_encode_tag_for_field(stream, field))
        return false;

    return pb_encode_string(stream, (uint8_t*)str, strlen(str));
}
int main()
{
FeatureFile featurefile = FeatureFile_init_zero;
pb_ostream_t stream = pb_ostream_from_buffer(buffer, sizeof(buffer));

featurefile.features.Id.funcs.encode = &encode_string;
}

但是在这个例子中,字符串“Hello world!” 是相当硬编码的xD,我怎样才能将字符串从main动态传递给该函数?

4

1 回答 1

8

pb_callback_t结构包含一个void* arg字段,您可以使用该字段通过arg参数将任何自定义数据传递给编码/解码函数。

在这种情况下,您可以这样做:

int main()
{
    ... 
    featurefile.features.Id.arg = "something";
    featurefile.features.Id.funcs.encode = &encode_string;
}

请注意,arg参数是指向 的指针void * const因此您将始终必须取消引用它:

bool encode_string(pb_ostream_t* stream, const pb_field_t* field, void* const* arg)
{
    const char* str = (const char*)(*arg);

    if (!pb_encode_tag_for_field(stream, field))
        return false;

    return pb_encode_string(stream, (uint8_t*)str, strlen(str));
}

请注意,您可以将指针传递给任何结构,即您可以轻松地创建一种“解析上下文”结构并将其传递,这样您就不需要关心解析函数将如何使用它。

在这种情况下,它可能类似于:

typedef struct
{
    const char * something;
    const char * whatever;
    ...
}
callback_context_t;

int main()
{
    callback_context_t ctx = { .something = "something" };

    // this way you always pass the same pointer type
    featurefile.features.Id.arg = &ctx;
    featurefile.features.Id.funcs.encode = &encode_string;
}

bool encode_string(pb_ostream_t* stream, const pb_field_t* field, void* const* arg)
{
    // ...and you always cast to the same pointer type, reducing
    // the chance of mistakes
    callback_context_t * ctx = (callback_context_t *)(*arg);

    if (!pb_encode_tag_for_field(stream, field))
        return false;

    return pb_encode_string(stream, (uint8_t*)ctx->something, strlen(ctx->something));
}
于 2019-08-20T08:40:09.253 回答