0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看威廉希尔官方网站 视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

C语言-函数的可变形参(不定形参)

DS小龙哥-嵌入式威廉希尔官方网站 来源:DS小龙哥-嵌入式威廉希尔官方网站 作者:DS小龙哥-嵌入式技 2022-08-14 09:58 次阅读

1. 前言

在学习C语言函数章节时发现,给函数传入的形参必须和函数定义原型的类型、数量一致才可以正常调用。

平时使用的printfscanf等函数时,传入的参数数量却可以随意改变,例如:

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

    文章

    7604

    浏览量

    136739
  • 函数
    +关注

    关注

    3

    文章

    4329

    浏览量

    62583
收藏 人收藏

    评论

    相关推荐

    C语言可变形是什么

      C语言允许定义参数数量可变函数,这称为可变参数函数(variadic function)。
    的头像 发表于 08-18 21:40 1327次阅读

    函数的实参与形

    函数定义中指定的形,在未发生函数调用时不占内存,只有函数调用时,函数中的形
    发表于 10-18 20:15

    单片机函数被改变的原因是什么?

    单片机函数被改变的原因是什么?
    发表于 10-19 07:03

    一种飞记录数据解析软件的设计与实现

    统的工作情况和各种飞行状态的信息。一旦浮空器发生飞行故障,飞记录数据为查找设备故障及失事原因提供重要依据。飞记录数据解析软件根据约定的传输协议,将特定形式的记录数据解析为便于查看的记录数据,其解析的效率直
    发表于 11-11 17:29 5次下载
    一种飞<b class='flag-5'>参</b>记录数据解析软件的设计与实现

    C++语言入门教程之C++语言程序设计函数的详细资料概述免费下载

    本文档的主要内容详细介绍的是C++语言入门教程之C++语言程序设计函数的详细资料概述免费下载内容包括了:1
    发表于 09-20 14:51 23次下载

    C语言函数不改变形内容的说明

    注:本文是作者以前发表在其个人博客,现在发布到“聚丰开发”专栏这也是面试时常碰到的题,通常是把一个指针作为函数的输入参数,在函数内部会改变输入参数对应的指针,问面试者在函数调用过程中指针的具体内容
    的头像 发表于 10-29 11:01 4973次阅读

    C语言总结_数组与函数练习题

    字符串标准处理函数介绍(string.h)、指针和数组当做函数,指针定义、函数返回指针、void类型定义指针、类型强制转换、常量声明、extern外边引用声明关键字。
    的头像 发表于 08-14 09:36 827次阅读

    C语言-函数的定义、声明、传

    C语言函数是非常重要的知识点,一个完整的C语言程序就是由主函数和各个子
    的头像 发表于 08-14 09:57 1942次阅读

    C语言-指针作为函数类型

    C语言函数里最常用就是指针传和返回地址,特别是字符串处理中,经常需要封装各种功能函数完成数据处理。
    的头像 发表于 08-14 10:05 1891次阅读

    数组/指针的传问题

    自定义函数里形的类型,要和函数调用中传过去的实参类型相对应
    的头像 发表于 08-17 10:37 1078次阅读

    关于征集《信息安全威廉希尔官方网站 散列函数》系列国家标准编单位的通知

    关于征集《信息安全威廉希尔官方网站 散列函数 第1部分 概述》标准编单位的通知 关于征集《信息安全威廉希尔官方网站 散列函数 第2部分 采用n位块密码的散列函数》标准
    的头像 发表于 11-15 14:39 675次阅读

    C语言函数调用的形式及过程

    C语言函数调用时的数据传递 在调用有函数时,主调函数和被调
    的头像 发表于 03-10 14:28 1747次阅读

    C语言函数参数介绍

    C语言数组元素作函数实参 数组元素可以用作函数实参,不能用作形C
    的头像 发表于 03-10 14:30 2382次阅读

    如何实现一个自己的printf函数代码?

    C语言中,可变参数函数可变参数宏都允许函数或宏接受不定
    发表于 09-06 14:20 980次阅读
    如何实现一个自己的printf<b class='flag-5'>函数</b>代码?

    Python函数的分类

    的分类 形的分类可以分为六种: 带有默认值的位置形 不带默认值的位置形 带有默认值的关键字形 不带默认值的关键字形
    的头像 发表于 11-14 11:19 821次阅读