前言

可变长参数指函数的参数个数在调用时才能确定的函数参数。基本上各种语言都支持可变长参数,在特定情形下,可变长参数使用起来非常方便。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. android 如何创建配置文件和读配置文件

    因为一些配置信息,多处用到的.且以后可能变更的,我想写个.prorperties配置文件给管理起来.在studio中新建一个Assets文件-->新建一个file文件类型为properties文 ...

  2. 分布式系统和CAP

    帽子理论(CAP): C:Consistency,一致性, 数据一致更新,所有数据变动都是同步的 A:Availability,可用性, 好的响应性能,完全的可用性指的是在任何故障模型下,服务都会在有 ...

  3. css3中的渐变小总结

      = 导航   顶部 线性渐变 径向渐变 透明度 边框 阴影   顶部 线性渐变 径向渐变 透明度 边框 阴影 系列教程 CSS3 Gradient分为linear-gradient(线性渐变)和r ...

  4. python 读写XLS

    需要库: xlrd, xlwt, xlutils 导入 import xlrd from xlutils.copy import copy 打开文件 data = xlrd.open_workbook ...

  5. Android-apktool反汇编异常-Input file (XXX) was not found or was not readable.

    问题描述: 原因: 在apptool 2.0以后命令发生了变化,不能再使用 apktool d <apk_path> <outfile_path> 解决方法:解决方法: 使用命 ...

  6. Project Euler:Problem 28 Number spiral diagonals

    Starting with the number 1 and moving to the right in a clockwise direction a 5 by 5 spiral is forme ...

  7. 动态加载Layout

    因为现在手头上做的需要显示很多不同布局,想着拆分开来不要全部都写到main.xml里,于是就想到动态加载Layout 目前试了下, LinearLayout page = (LinearLayout) ...

  8. hibernate validator 专题

    JSR-303 原生支持的限制有如下几种 : 限制 说明 @Null 限制只能为 null @NotNull 限制必须不为 null @AssertFalse 限制必须为 false @AssertT ...

  9. 机器学习: TensorFlow with MLP 笑脸识别

    Tensor Flow 是一个采用数据流图(data flow graphs),用于数值计算的开源软件库.节点(Nodes)在图中表示数学操作,图中的线(edges)则表示在节点间相互联系的多维数据数 ...

  10. matlab 三维图像的绘制

    1. 基本绘图函数 plot3() scatter3() 2. 修饰与点缀 但仅使用默认的配置,调用这些绘图函数,不会很丑,但也漂亮不到哪里去. view(az, el),调整视野和视角: