【发布时间】:2020-02-04 10:57:07
【问题描述】:
当试图编译这段代码时
#include <stdarg.h>
void bar_ptr(int n, va_list *pvl) {
// do va_arg stuff here
}
void bar(int n, va_list vl) {
va_list *pvl = &vl; // error here
bar_ptr(n, pvl);
}
void foo(int n, ...) {
va_list vl;
va_list *pvl = &vl; // fine here
va_start(vl, n);
bar(n, vl);
va_end(vl);
}
int main() {
foo(3, 1, 2, 3);
return 0;
}
GCC 编译器在 bar 函数中打印关于 initialization from incompatible pointer type 的警告。在foo 中也可以使用相同的语句。
似乎va_list 类型的参数的类型不是va_list。这可以使用静态断言轻松测试,例如
_Static_assert(sizeof(vl) == sizeof(va_list), "invalid type");
在bar 函数中。使用 GCC,_Static_assert 失败。同样可以在 C++ 中使用 declytpe 和 std::is_same 进行测试。
我想获取bar 的va_list vl 参数的地址,并将其作为bar_ptr 的参数传递,以像thread 中描述的那样进行思考。另一方面,可以直接从main 调用bar_ptr(n, pvl),替换bar(n, vl)。
根据C11 final draft的脚注253,
允许创建指向
va_list的指针并将其传递给 指向另一个函数的指针
如果 va_list 被定义为函数的参数,而不是在函数体中,为什么不能这样做?
解决方法:
即使这不能回答问题,一个可能的解决方法是使用 va_copy 创建的参数的本地副本来更改 bar 的内容:
void bar(int n, va_list vl) {
va_list vl_copy;
va_copy(vl_copy, vl);
va_list *pvl = &vl_copy; // now fine here
bar_ptr(n, pvl);
va_end(va_copy);
}
【问题讨论】:
-
关于您拥有的标准引用:您实际上并没有将指针传递给
va_list。 -
允许创建指向 va_list 的指针并将该指针传递给另一个函数您没有将 指针 传递给 @987654347 @,你传递的是一个实际的
va_list。 -
我知道我正在传递一个
va_list。假设我有第三个函数void bar_ptr(va_list *pvl);,我想将指向va_list vl的指针传递给该函数。我将编辑问题以指定它。 -
您拥有的
va_copy方法是解决此问题的规范方法。我有几个过去的问题基本上得出了这个结论。
标签: c variadic-functions