类型int,是我的泛型类型>要了解,请持续阅读.
所以我的问题是:
void function (int paramN,...);
在我的程序中,没有办法知道,这是变量参数的类型,它可以是一个char,一个数组,一个int,一个short,一个函数点等…
function (paramN,"Hey,I'm a string",1,function_pint,array,-1); // -1 is a sentinel.
所以,我认为,一个int是32位,在x86(32位)系统中,这将保存所有的内存地址.所以,如果我使用int获得所有参数,这不会是一个问题,例如,“嘿,我是一个字符串”这个字符串的地址,通常适合一个32位变量,所以我只需要做一个演员
我是对的?
我可以做吗?
注意:我不想让我的功能像printf(这个解决方案,不适合在这种情况下吗?)
感谢所有答案.
对不起,我的英文不好
解决方法
C调用约定是为了调用者在堆栈上放置参数,但它不会在类型上放置任何信息,因此被调用方必须有一种方法来找到它(至少是变量的大小).
>每个类型的原型函数都没有问题.
>使用变量或参数(可变)的函数更加棘手,您必须为每个参数调用va_arg来读取每个变量,并且必须提供va_arg的类型.如果您提供的类型不是编译器不会投诉的类型(它不能是运行时信息),但是任何事情都可能发生(通常是坏的).
因此你必须通过类型.
在某些情况下,您可以预测类型(例如:一个计数器后跟一些int,等等),但通常会将其传递给参数编码.你可以传递它像格式字符串像printf那样编码,由jldupont描述的联合技巧也很常见.
但无论如何你必须通过它.
你真的不能依赖底层数据二进制表示,甚至不能数据大小.即使程序在写入时似乎工作,它也没有兼容性,并且可以随系统,编译器的任何变化,甚至在更改编译选项时发生变化.
我们来看一个例子,你传递类型后跟参数(既不是联合技巧也不是像printf的格式字符串).它所做的是将所有将其传递的值转换为双倍并添加它们,没有真正有用的不是它:
#include <stdio.h> #include <stdarg.h> enum mytypes {LONG,INT,FLOAT,DOUBLE }; double myfunc(int count,...){ long tmp_l; int tmp_i; double tmp_d; double res = 0; int i; va_list ap; va_start(ap,count); for(i=0 ; i < count; i++){ int type = va_arg(ap,enum mytypes); switch (type){ case LONG: tmp_l = va_arg(ap,long); res += tmp_l; break; case INT: tmp_i = va_arg(ap,int); res += tmp_i; break; case FLOAT: /* float is automatically promoted to double when passed to va_arg */ case DOUBLE: tmp_d = va_arg(ap,double); res += tmp_d; break; default: /* unknown type */ break; } } va_end(ap); return res; } int main(){ double res; res = myfunc(5,LONG,(long)1,(int)10,DOUBLE,(double)2.5,(double)0.1,(float)0.3); printf("res = %f\n",res); }
这个例子使用C99中定义的新的stdarg可变标题.要使用它,您需要至少有一个固定参数到您的函数(在这个例子中它的计数).好的,如果这样,你可以在你的函数中有几个可变列表(例如像myfunc(int count1,…,int count2,…)).不好的是,你不能有一个纯粹的可变功能(即像myfunc(…)这样的东西,就像旧的格式一样,你仍然可以使用旧的格式,使用varargs兼容性头,但更复杂,很少有必要,因为你需要类型,但也有一些方式来知道列表完成,像count这样的东西是方便的(但不是唯一的办法,例如可以使用’终止符’).