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

简介: 平时使用的printf ,scanf等函数时,发现它们传入的参数数量可以随意改变,这篇文章就是介绍它们是如何实现的。

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 <stdio.h>
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 <stdio.h>
int scanf(const char *format, ...);
int fscanf(FILE *stream, const char *format, ...);
int sscanf(const char *str, const char *format, ...);

发现这些函数定义时,参数列表里有一个省略符号...,这个省略符号就表示当前函数支持不定长形参

示例代码:可变形参的声明方式

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
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 <stdarg.h>
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 <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdarg.h>

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 <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdarg.h>

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 <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdarg.h>

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 <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdarg.h>

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
}
目录
相关文章
|
8天前
|
存储 Serverless C语言
【C语言基础考研向】11 gets函数与puts函数及str系列字符串操作函数
本文介绍了C语言中的`gets`和`puts`函数,`gets`用于从标准输入读取字符串直至换行符,并自动添加字符串结束标志`\0`。`puts`则用于向标准输出打印字符串并自动换行。此外,文章还详细讲解了`str`系列字符串操作函数,包括统计字符串长度的`strlen`、复制字符串的`strcpy`、比较字符串的`strcmp`以及拼接字符串的`strcat`。通过示例代码展示了这些函数的具体应用及注意事项。
|
11天前
|
存储 C语言
C语言程序设计核心详解 第十章:位运算和c语言文件操作详解_文件操作函数
本文详细介绍了C语言中的位运算和文件操作。位运算包括按位与、或、异或、取反、左移和右移等六种运算符及其复合赋值运算符,每种运算符的功能和应用场景都有具体说明。文件操作部分则涵盖了文件的概念、分类、文件类型指针、文件的打开与关闭、读写操作及当前读写位置的调整等内容,提供了丰富的示例帮助理解。通过对本文的学习,读者可以全面掌握C语言中的位运算和文件处理技术。
|
11天前
|
存储 C语言
C语言程序设计核心详解 第七章 函数和预编译命令
本章介绍C语言中的函数定义与使用,以及预编译命令。主要内容包括函数的定义格式、调用方式和示例分析。C程序结构分为`main()`单框架或多子函数框架。函数不能嵌套定义但可互相调用。变量具有类型、作用范围和存储类别三种属性,其中作用范围分为局部和全局。预编译命令包括文件包含和宏定义,宏定义分为无参和带参两种形式。此外,还介绍了变量的存储类别及其特点。通过实例详细解析了函数调用过程及宏定义的应用。
|
17天前
|
Linux C语言
C语言 多进程编程(三)信号处理方式和自定义处理函数
本文详细介绍了Linux系统中进程间通信的关键机制——信号。首先解释了信号作为一种异步通知机制的特点及其主要来源,接着列举了常见的信号类型及其定义。文章进一步探讨了信号的处理流程和Linux中处理信号的方式,包括忽略信号、捕捉信号以及执行默认操作。此外,通过具体示例演示了如何创建子进程并通过信号进行控制。最后,讲解了如何通过`signal`函数自定义信号处理函数,并提供了完整的示例代码,展示了父子进程之间通过信号进行通信的过程。
|
17天前
|
C语言
C语言 字符串操作函数
本文档详细介绍了多个常用的字符串操作函数,包括 `strlen`、`strcpy`、`strncpy`、`strcat`、`strncat`、`strcmp`、`strncpy`、`sprintf`、`itoa`、`strchr`、`strspn`、`strcspn`、`strstr` 和 `strtok`。每个函数均提供了语法说明、参数解释、返回值描述及示例代码。此外,还给出了部分函数的自实现版本,帮助读者深入理解其工作原理。通过这些函数,可以轻松地进行字符串长度计算、复制、连接、比较等操作。
|
18天前
|
SQL 关系型数据库 C语言
PostgreSQL SQL扩展 ---- C语言函数(三)
可以用C(或者与C兼容,比如C++)语言编写用户自定义函数(User-defined functions)。这些函数被编译到动态可加载目标文件(也称为共享库)中并被守护进程加载到服务中。“C语言函数”与“内部函数”的区别就在于动态加载这个特性,二者的实际编码约定本质上是相同的(因此,标准的内部函数库为用户自定义C语言函数提供了丰富的示例代码)
|
1月前
|
C语言
【C语言】字符串及其函数速览
【C语言】字符串及其函数速览
23 4
|
1月前
|
编译器 程序员 C语言
【C语言篇】从零带你全面了解函数(包括隐式声明等)(下篇)
⼀般情况下,企业中我们写代码时候,代码可能⽐较多,不会将所有的代码都放在⼀个⽂件中;我们往往会根据程序的功能,将代码拆分放在多个⽂件中。
|
1月前
|
测试技术 C语言
C语言中的void函数
C语言中的void函数
|
1月前
|
存储 安全 编译器
C语言中的scanf函数
C语言中的scanf函数