我想调用一个extern "C"
函数,例如f1(int a, float f, double d, void* ptr)
使用带有实际参数的前向声明,但是在实际实现中,我想使用argsva_list
和它的朋友pop
。让我们想象一下我有一个有效的用例,是否可以使用?
main.cpp
extern "C" void f(int anchor, int a, float f, double d, void* ptr, char c);
int main(int, char*)
{
f(0, 42, 1.08f, 3.14, reinterpret_cast<void*>(0xcafebabe), 'c');
return 0;
}
impl.cpp
#include <cstdarg>
#include <iostream>
#include <iomanip>
using namespace std;
void f(int anchor, ...)
{
va_list args;
va_start(args, anchor);
int a = va_arg(args, int);
float f = va_arg(args, float);
double d = va_arg(args, double);
void* ptr = va_arg(args, void*);
char c = va_arg(args, char);
cout << a << ' '
<< f << ' '
<< d << ' '
<< hex << (std::ptrdiff_t)ptr << ' '
<< (int)c << endl;
va_end(args);
}
该代码至少会在MSVC 2015上运行并打印正确的值,现在的问题是:它是否可以保证工作,如果不能,那么,它是否仍可以在最重要的平台和编译器上工作?
从理论上讲,您的代码会受到不确定的行为的影响。实际上,当我尝试使用g ++构建后运行程序时,我得到了
Illegal instruction (core dumped)
原因是当函数具有可变参数参数时,它只能处理int
s,double
s,指针。它不能处理float
s和char
s。
声明是否指定为:
extern "C" void f(int anchor, ...);
在float
将被提升到double
和char
将被提升到一个int
。
在实现方面,您需要使用:
float f = (float)va_arg(args, double);
和
char c = (char)va_arg(args, int);
有关该主题的更多信息,请参见C函数调用中的默认参数提升。
本文收集自互联网,转载请注明来源。
如有侵权,请联系[email protected] 删除。
我来说两句