前言

可变长参数指函数的参数个数在调用时才能确定的函数参数。基本上各种语言都支持可变长参数,在特定情形下,可变长参数使用起来非常方便。c语言中函数可变长参数使用“...”来表示,同时可变长参数只能位于固定参数的后面,固定参数的个数至少为1。只要学习过c语言的,应该都知道printf函数,并且见识到了其强大的功能——事实上,迄今为止,我仍认为这是c函数库中最牛逼的函数之一。

一、一个简单的例子

  1. #include <string> 
  2. #include <stdio> 
  3. ///拼接字符串 
  4. char * JointStr(int Count, ...)
  5.   char * pszBuff = new char[100]; 
  6.   ::memset(pszBuff, 0, 100);   
  7.   va_list vl;  
  8.   va_start(vl, Count); 
  9.   for(int i = 0; i < Count; i++) 
  10.   { 
  11.       strcat(pszBuff, va_arg(vl, char *)); 
  12.   }   
  13.   return pszBuff; 
  14. void main() 
  15.    char * pszStr = JointStr(3, "abc", "123", "!@#"); 
  16.    printf("%s", pszStr); 
  17. }

  执行后,输出:abc123!@#

函数JointStr的功能是指定个数的字符串拼接起来,返回拼接后的字符串的指针。参数Count是字符串的个数,后面跟可变长参数,使用时应该跟Count个char*型参数。使用时,可以随意指定个数。(该例子只是用来说明问题,实际使用时不会用这个函数,可以使用标准库函数中的sprintf函数)。

二、可变长参数的使用方法

首先,必须弄清楚一下三个宏定义:

  • va_arg
  • va_start
  • va_end

以及一个类型:va_list

从c函数库中的头文件中可以看到va_list的定义:

typedef char *  va_list;

也就是说它就是一个指针,那么该指针指向什么地方呢? 这就是va_start的作用了。首先看看这三个宏定义的申明:

  1. #define va_start(ap,v) 
  2. #define va_arg(ap,t) 
  3. #define va_end(ap)

va_start有两个参数:第一个ap应该填写va_list,第二个v应该填写函数参数列表中(可以认为传给函数的参数是一个列表,一个接一个)的某个参数,例如例子中的Count。其作用是将ap指向函数参数列表中的参数v的位置(msdn是这样说的,我觉得应该指向参数列表中的参数v的下一个参数的开始地址)。

va_end作用是将ap设置为NULL。

va_arg(ap,t)有两个参数:第一个是va_list,第二个是参数类型。其作用是从ap开始取一个t型的值返回,并且自动将ap指向下一个参数。所以如果t即参数类型写错了,例如将char*写成char了,本来要取4个字节,结果只取了一个字节,ap本来要向后面移动4个字节,结果只移动了一个字节,后面的数据就全错了。同时,如果你多取了一次参数,将报内存越界错误,所以使用可变长参数,前面一般都会传一个参数来指定参数的个数。

使用可变长参数的步骤:

  1. 声明va_list变量
  2. 使用va_start指定可变长参数的位置
  3. 使用va_arg来获取参数值
  4. 可选,使用va_end将va_list清零

三、va_list类型作为参数

在c标准库中有一个函数vsprintf,声明如下:

int __cdecl vsprintf(char *_DstBuf, char * _Format, va_list _ArgList);

其第三个参数为va_list类型,我们可以在这样使用:

char* GetFormatStr(char* format, ...)

{
  char *pszBuff = new char[256];
  va_list vl;
  va_start(vl,format);
  vsprintf(pszBuff, format, vl);
  return pszBuff;
}
void main()
{
  char * str = GetFormatStr("%s is %d year old.", "frank", 25);
  printf("%s", str);
}

将输出:frank is 25 year old.

结束语

我们可以认为,我们调用可变长参数的函数时,传递给函数的参数是在堆栈中保存为一个紧挨一个的列表。获取参数的唯一方法就是通过参数列表的指针,取一个参数,移动一个参数长度的指针,实际上如何取参数完全掌握在用户手中,用户应当小心应对。

原文链接:

https://blog.csdn.net/frank_liuxing/article/details/18000825

关于C中可变长参数的更多相关文章

  1. java中可变长参数

    ** * Created by Lenovo on 2017/12/10. * java中可变长参数 */ public class reflect04 { //m1有一个int类型的可比变长参数 / ...

  2. Java中可变长参数的使用及注意事项

    在Java5 中提供了变长参数(varargs),也就是在方法定义中可以使用个数不确定的参数,对于同一方法可以使用不同个数的参数调用,例如print("hello");print( ...

  3. Java中可变长参数的方法

    原文转自:http://www.cnblogs.com/lanxuezaipiao/p/3190673.html 在Java5 中提供了变长参数(varargs),也就是在方法定义中可以使用个数不确定 ...

  4. java中可变长参数的定义及使用方法

    JAVA中可以为方法定义可变长参数( Varargs)来匹配不确定数量的多个参数,其定义用“...”表示.其实,这类似于为方法传了一个数组,且在使用方法上也和数组相同,如下: public void ...

  5. 在Python中使用可变长参数列表

    函数定义 在函数定义中使用*args和**kwargs传递可变长参数. *args用作传递非命名键值可变长参数列表(位置参数); **kwargs用作传递键值可变长参数列表 函数调用 在调用函数时,使 ...

  6. Python中函数的参数传递与可变长参数

    转自旭东的博客原文 Python中函数的参数传递与可变长参数 Python中传递参数有以下几种类型: (1)像C++一样的默认缺省函数 (2)根据参数名传参数 (3)可变长度参数 示例如下: (1)默 ...

  7. Java中的可变长参数

    可变长参数的定义 与一般方法没多大差别,只不过形参多了...(三个点) 方法名(数据类型 ... 变量名){} 小案例: public class ParamDemo { public static ...

  8. javascript arguments解释,实现可变长参数。

    在C#中,有可变长参数params[],但是在js中,如何实现这种可变参数呢? 一.可变长参数 arguments是非常好的解决方法,一直不知道javascript有这个东西. 先来看看应用场景,使用 ...

  9. python 函数可变长参数

    python中的可变长参数有两种: 一种是非关键字参数(*元组),另一种是关键字参数(**字典) 非关键字可变长参数: """ 非关键字可变参数,一个星号作为元组传入函数 ...

随机推荐

  1. Java中利用Math.random()产生服从泊松分布的随机数

    众所周知.Java的Math.random()产生的是服从均匀分布的随机数,可是其它分布的应用也相当广泛,比如泊松分布和高斯分布(正态分布).而这些分布Java没有非常好的提供(高斯分布能够利用Ran ...

  2. Windows获取时间函数(使用GetLocalTime,GetSystemTime,SystemTimeToTzSpecificLocalTime,GetFileTime API函数

    获取本地时间 typedef struct _SYSTEMTIME { WORD wYear; WORD wMonth; WORD wDayOfWeek; WORD wDay; WORD wHour; ...

  3. Mina、Netty、Twisted一起学习(三):TCP前缀固定大小的消息(Header)

    于以前的博文于,有介绍切割消息换行的方法. 但是有一个小问题,这样的方法,设消息中本身就包括换行符,那将会将这条消息切割成两条.结果就不正确了. 本文介绍第二种消息切割方式,即上一篇博文中讲的第2条: ...

  4. 单点登录原理与简单实现--good

    一.单系统登录机制 1.http无状态协议 web应用采用browser/server架构,http作为通信协议.http是无状态协议,浏览器的每一次请求,服务器会独立处理,不与之前或之后的请求产生关 ...

  5. win10 uwp 线程池

    原文:win10 uwp 线程池 如果大家有开发 WPF 或以前的程序,大概知道线程池不是 UWP 创造的,实际上在很多技术都用到线程池. 为什么需要线程池,他是什么?如何在 UWP 使用线程池,本文 ...

  6. 把搜狗输入法词库导入Google拼音输入法

    为PC端Google拼音输入法增加词库 为什么折腾词库 都在说百度.讯飞等输入法上传用户词库,为了安全建议大家使用google输入法之类,话说回来,要想使用智能联想功能是不是就得把你输入习惯放在他的里 ...

  7. Qt自定义密码框,先显示后隐藏(继承以后改写slot即可,即与哪个相近就改写哪个)good

    现在很多应用在密码输入时,会先显示一段时间,大概几百毫秒,然后再变成星号或者圆点隐藏起来.这样做的好处是,可以让密码输入者看到自己输入的字符,同时又防止密码被偷窥.但是Qt自带的密码输入框,要么输入时 ...

  8. [UWP]在应用开发中安全使用文件资源

    原文:[UWP]在应用开发中安全使用文件资源 在WPF或者UWP应用开发中,有时候会不可避免的需要操作文件系统(创建文件/目录),这时候有几个坑是需要大家注意下的. 创建文件或目录时的非法字符检测 在 ...

  9. C# 实现生成带二维码的专属微信公众号推广海报

    原文:C# 实现生成带二维码的专属微信公众号推广海报 很多微信公众号中需要生成推广海报的功能,粉丝获得专属海报后可以分享到朋友圈或发给朋友,为公众号代言邀请好友即可获取奖励的.海报自带渠道二维码,粉丝 ...

  10. Android TV开发中所有的遥控器按键监听及注意事项,新增home键监听

    原文:Android TV开发中所有的遥控器按键监听及注意事项,新增home键监听 简单记录下android 盒子开发遥控器的监听 ,希望能帮到新入门的朋友们 不多说,直接贴代码 public cla ...