1. 前言
在学习C语言函数章节时发现,给函数传入的形参必须和函数定义原型的类型、数量一致才可以正常调用。
平时使用的printf
,scanf
等函数时,传入的参数数量却可以随意改变,例如:
printf("大家好");
printf("我是整数:%d\n",123);
printf("%d%d%d%d\n",1,2,3,4);
printf("%s%s%s\n","1","2","3","4");
printf
函数是如何实现这种传参方式的?
我们看一下printf,scanf
系列函数的原型。
#include
int printf(const char *format, ...);
int fprintf(FILE *stream, const char *format, ...);
int sprintf(char *str, const char *format, ...);
int snprintf(char *str, size_t size, const char *format, ...);
#include
int scanf(const char *format, ...);
int fscanf(FILE *stream, const char *format, ...);
int sscanf(const char *str, const char *format, ...);
发现这些函数定义时,参数列表里有一个省略符号...
,这个省略符号就表示当前函数支持不定长形参
。
示例代码:可变形参的声明方式
#include
#include
#include
void func(char *p,...);
int main(int argc,char **argv)
{
func("123",1,2,3,4,"",12.345);
return 0;
}
//正确的
void func(char *p,...)
{
}
//错误的
void func2(...,char *p)
{
}
//错误的
void func3(...)
{
}
2. 可变形参本身实现原理
明白了如何定义可变形参,接下来就得学习可变形参的原理,然后学习如何去提取这些传入的参数。
(1). 函数的形参是放在栈空间的。
(2). 可变形参,传入的多余的参数都是存放在栈空间。
存放内存地址是连续的。
理论上只要知道传入参数的首地址,就可以推出其他参数的地址。
系统的标准参数头文件和处理可变形参的相关函数
#include
int vprintf(const char *format, va_list ap);
int vfprintf(FILE *stream, const char *format, va_list ap);
int vsprintf(char *str, const char *format, va_list ap);
int vsnprintf(char *str, size_t size, const char *format,va_list ap);
直接查看头文件的帮助:
[wbyq@wbyq linux_c]$ man stdarg.h
void va_start(va_list ap, argN); //开始
void va_copy(va_list dest, va_list src); //拷贝
type va_arg(va_list ap, type); //取具体形参—取值
void va_end(va_list ap); //结束
va_list ap; 就是定义一个char类型的指针。va_list==char *
3. 单独提取参数列表里的值
#include
#include
#include
#include
void foo(char *fmt, ...);
int main(int argc,char **argv)
{
foo("%d,%s,%c",12,"123",'A');
return 0;
}
// foo("%d,%s,%c",12,"123",'A')
void foo(char *fmt, ...)
{
va_list ap; //定义一个char类型指针
int d;
char c, *s;
va_start(ap, fmt); //指针地址赋值--初始化
while (*fmt)
switch (*fmt++) {
case 's': /* string */
s = va_arg(ap, char *);
printf("string %s\n", s);
break;
case 'd': /* int */
d = va_arg(ap, int);
printf("int %d\n", d);
break;
case 'c': /* char */
c = (char) va_arg(ap, int);
printf("char %c\n", c);
break;
}
va_end(ap); //将ap指针置为NULL
}
4. 使用格式化方式提取形参列表里的值
#include
#include
#include
#include
void foo(char *fmt, ...);
int main(int argc,char **argv)
{
foo("int=%d,string=%s char=%c",12,"123",'A');
return 0;
}
// foo("%d,%s,%c",12,"123",'A')
void foo(char *fmt, ...)
{
char buff[100];
va_list ap; //定义一个char类型指针
va_start(ap, fmt); //指针地址赋值--初始化
//将参数列表里所有参数,按照格式化转换成字符串-存放到str指向的空间
vsprintf(buff,fmt,ap);
va_end(ap); //将ap指针置为NULL
printf("%s\n",buff);
}
5. 提取可变形参列表里的单个数据
#include
#include
#include
#include
void foo(char *fmt, ...);
int main(int argc,char **argv)
{
foo("sdcf","hello",666,'A',123.456);
return 0;
}
void foo(char *fmt, ...)
{
va_list ap; //定义一个char类型指针
int d;
char c, *s;
double f;
va_start(ap, fmt); //指针地址赋值--初始化
while(*fmt) //遍历fmt指针指向空间的值
{
switch(*fmt++)
{
case 's': /* string */
s = va_arg(ap, char *);
printf("字符串:%s\n", s);
break;
case 'd': /* int */
d = va_arg(ap, int);
printf("整型:%d\n", d);
break;
case 'c': /* char */
c = (char) va_arg(ap,int);
printf("字符:%c\n", c);
break;
case 'f': /* float */
f = va_arg(ap, double);
printf("浮点数:%f\n", f);
break;
}
}
va_end(ap); //将ap指针置为NULL
}
6. 精简代码-提取可变形参列表里的单个数据
#include
#include
#include
#include
void foo(char *fmt, ...);
int main(int argc,char **argv)
{
foo("123","hello",666,'A',123.456);
return 0;
}
void foo(char *fmt, ...)
{
va_list ap; //定义一个char类型指针
va_start(ap, fmt); //指针地址赋值--初始化
printf("第一个字符串:%s\n",fmt);
printf("提取字符串:%s\n",va_arg(ap,char*));
printf("提取整数:%d\n",va_arg(ap,int));
printf("提取字符:%c\n",va_arg(ap,int));
printf("提取字符:%lf\n",va_arg(ap,double));
va_end(ap); //将ap指针置为NULL
}
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。
举报投诉
-
C语言
+关注
关注
180文章
7521浏览量
127407 -
函数
+关注
关注
3文章
3859浏览量
61296
发布评论请先 登录
相关推荐
C语言——可变参数问题.
;The value is %d!\n", value); 这种可变参数可以说是C语言一个比较难理解的部分,这里会由几个问题引发一些对它的分析。 注意:在C++中有
发表于 04-20 15:17
函数的形参,在使用函数时,为什么为变化?
例如:void delay(uint z){uint x, y;for(x = z; x > 0; x--)for(y = 110; y > 0; y--);}我在调试时,观察X和Z的变化情况,发现z和x的值相同,然后做相同的变化。为什么形参也会发生变化?
发表于 08-19 14:13
stm32库函数形参查错功能
的STM32库函数中,对于其形参提供了查错机制。如下示例是我们常用的GPIO初始化函数,在函数的一开始就使用assert_param(x)对形参
发表于 09-11 10:09
函数形参为volatile类型,这个有什么意义?
((HAL_GetTick() - tickstart) < Delay){}}这是用STM32CubeMX生成的带操作系统工程里面的一个函数,如何理解形参Delay前面的__IO,还望各路高手指导。
发表于 11-23 16:44
PID调参的相关资料分享
说明:本文章适用于STM32初学者,想完成一个好玩且有深度的项目但不知道从何下手的同学。PID调参是平衡车的精髓所在,参数整定的好坏直接影响到平衡车的平衡效果。有的车平衡时来回晃而参数选的好的车就能
发表于 01-14 09:14
如何实现一个可变参的“printf”函数
目录标准库测试用过stm32的人应该在调试串口的时候都用过重定义的printf它可以和c语言中的printf一样的使用。如何实现这么一个函数呢?标准库#include <
发表于 01-24 06:58
评论