xml-rpc-c学习有感—c语言不定参函数的实现

博客探讨了C语言中如何通过xml-rpc-c库处理不定参数函数,重点在于va_start、va_arg和va_end等宏的使用,以及'(ii)'在标识参数类型中的作用。通过引用相关文献,解释了如何利用这些宏来获取和处理参数。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

     最近在看xmlprc-c的demo,突然发现在xmlrpc_client_call()
xmlrpc_client_call(xmlrpc_env * const envP,
                   const char * const server_url,
                   const char * const method_name,
                   const char * const format,
                   ...);
函数中有一个参数“format”demo中使用传参方式为
xmlrpc_client_call(&env, serverUrl, methodName,
                                 "(ii)", (xmlrpc_int32) 5, (xmlrpc_int32) 7);

怪哉为何使用"(ii)",深究下去,发现这和C语言不定参数函数的传参有关,由于某些函数不能确定参数的个数(如printf),而参数在内存中得表示方法为堆栈,最后的参数最后入栈,因此可以记录最后一个固定参数(当然可以用&取得地址),然后根据user指定的每个参数的类型,使用指针偏移计算,便可取得所有的参数,参考文献如下:

www.cppblog.com/qiujian5628/archive/2008/01/21/41562.html

http://archive.cnblogs.com/a/2299868/

以上文章对此讲的已经比较详细了,留心不难发现,最重要的就是下面几个宏定义,vc头文件中stdarg.h定义如下:

#define _INTSIZEOF(n)   ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )

#define va_start(ap,v)  ( ap = (va_list)&v + _INTSIZEOF(v) )
#define va_arg(ap,t)    ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
#define va_end(ap)      ( ap = (va_list)0 )

它们作用分别如下

v——最后一个固定参数

t——当前参数的类型

ap——当前参数指针

va_start——计算第一个不定参的地址,有了地址,我们就可以一个一个的取出来;

va_arg——返回当前参数,并将ap移动到下一个参数;

va_end——很简单,ap置为空,表示取参结束;

不得不说写这些代码的作者确实牛X,av_start比较好理解,难点在va_arg定义,乍一看貌似这个宏貌似什么都没干,但仔细看来,却大有文章。首先,我们需要明确一点,宏定义=代码替换,也就是说在出现va_arg()的地方都应该直接替换成

 ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
若令 m = va_arg(ap, t); 得到的m当然是当前参数,而同时ap也移动到下一个参数的地址;这里,ap+=_INTSIZEOF(t)实现了后者,注意,ap += XXX - XXX 并不等价于 ap, 当然,从返回值上来是等价的,但对ap来说,对它起作用的只有ap += XXX,这是问题的关键(作用类似于ap++);

于是这段简单的代码变实现了一石二鸟的作用:1,返回当前参数的值;2,ap移到下一个参数地址;

最后,"(ii)"的定义属于xml-rpc-c的范畴,i代表int,相当于告诉xml-rpc库user各个参数的类型,这是必要的,要不然便无从取参了,具体参见

static void
getValue(xmlrpc_env *    const envP, 
         const char**    const formatP,
         va_listx *      const argsP,
         xmlrpc_value ** const valPP)

完整定义在  xmlrpc_build.c中可以找到。






                
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值