C和C++中的不定参数

在初学C的时候,我们都会用到printf函数来写Hello World的程序.在我们看printf函数的声明时,会看到类似于下面代码

int     printf(const char * __restrict, ...);

另外,在我们学习C和C++的时候,函数的声明总是确定个数和类型的,而我们在用printf的时候,却可以一次输出多个参数.

这就是我们要提的不定参数了.

在32位的C和C++编程中,函数调用是有规约的,并且各个编译器也基本达成了一致,尽管他们编译出的东西基本不能通用.关于调用规约的东西,可以参考一下维基百科.

在64位的C和C++编程中,就没有调用规约的概念了,基本上做到了统一,但是不同的编译器的传参方式却不尽相同,这里我们不去讨论了,有兴趣的可以编译到汇编代码查看.

不定参数的函数调用方式为cdec方式的,也就是由调用者来恢复参数栈,这个不难理解,因为被调用的函数无从得知有多少个参数传进来,所以不可能知道如何恢复栈.

如果上面写的你看不懂,不要紧,你可以用google搜一下,相信很快你就会明白了,不搜也不要紧.

在使用不定参数时,我们会用到三个宏,分别是

va_start , va_arg 和 va_end

还有一个类型 va_list

它们都定义在 stdarg.h 或者 cstdarg (C++)里,使用时记得引入.

其中,va_start用来用于初始化va_list, va_arg用来读取va_list中的参数,当所有的读取都结束后要用va_end来释放va_list.

下面写一个示例

#include<stdlib.h>
#include<stdio.h>
#include<stdarg.h>
int sum(int count, ...);
int main()
{
    int nS = sum(3,1,2,3);

    printf("%d\n", nS);
    return 0;
}
int sum(int count, ...)
{
    int _sum,arg,i;
    va_list arg_ptr;
    _sum = 0;
    va_start(arg_ptr,count);
    for(i=0; i < count; ++i)
    {
        arg = va_arg(arg_ptr,int);
        _sum += arg;
    }
    va_end(arg_ptr);
    return _sum;
}

上面的例子比较简单,后续参数个数由第一个参数指定,而且类型默认都是int类型的.

在printf函数中,后续参数个数是由第一个格式字符串来指定的,并且指定了参数类型,比如%d 说明对应的参数是整形而%f 对应的是浮点类型.

接下来我们看看这几个宏.

va_start是用来初始化va_list的,第一个参数是参数表的指针,第二个参数是不定参数前的最后一个参数.

va_arg 是用来读取不定参数,第一个参数是参数表的指针,第二个参数是参数的类型.函数本身并不知道参数的类型,所以使用不当会导致出错.

va_end是用来释放va_list占用的资源的,只有一个参数,就是要释放的va_list.

最后,我们通过printf函数来总结一下使用不定参数的一些规范:

1. 函数本身必须有办法知道不定参数的类型,比如printf通过格式化字符串通知函数后续参数的每个参数的类型,其中的%d类形的格式与后续的参数是一一对应的.在我写的示例代码中,是默认约定了所有参数都是int 类型的.

2. 函数必须能知道参数结束的地方,printf函数是由格式输出字符串来知道的,当没有类似%d或%f 这种字符出现时,参数就结束了.我给出的例子中,是通过第一个参数给定了后续参数的个数的.

3.调用必须严格按照调用的约定来做,而且不定的参数是不会自动转型的,比如当我们 printf("%d",3.3) 会发现输出的不是3,就是因为3.3作为一个浮点数传入,而不会因为格式字符串中的%d自动转成整数.要想得到预期的结果,需要写成下面这样 printf("%d",(int)3.3)

时间: 2024-12-15 09:54:02

C和C++中的不定参数的相关文章

JavaScript中使用arguments来实现不定参数的计算

在JavaScript中可以使用arguments对象来实现不定参数 例如:不定参数的相加实现 add(1,2)      ==>3 add(1,2,3)   ==>6 var add = function(){ var total = 0; for(var index = 0;index < arguments.length; index++){ total = total + arguments[index]; } return total; } alert("1加2的值是

golang中不定参数与数组切片的区别

package main import "fmt" func main() { myfunc1(88, 42, 12, 56) //传递不定数量的参数 myfunc2([]int{88, 42, 12, 56}) //传递一个数组切片 } func myfunc1(args ...int) { //接受不定数量的参数,这些参数的类型全部是int for _, arg := range args { fmt.Println(arg) } } func myfunc2(args []int

java的不定参数,可变参数Object ...

对于java中不定参数,或者是可变参数Object ... 这是JDK1.5的新特性 针对不定参数的其他问题,可以参考Java不定长度参数[http://blog.chinaunix.net/uid-200142-id-3124427.html]

不定参数的应用

不定参数当年做为C/C++语言一个特长被很多人推崇,但是实际上这种技术并没有应用很多.除了格式化输出之外,我实在没看到多少应用.主要原因是这种技术比较麻烦,副作用也比较多,而一般情况下重载函数也足以替换它.尽管如此,既然大家对它比较感兴趣,我就简单总结一下它的使用和需要注意的常见问题.原理:刚学C语言的时候,一般人都会首先接触printf函数.通过这个函数,你可以打印不定个数的变量到屏幕,如:printf("%d", 3);printf("%d,%d",3,4);

JDK5不定参数方法

jdk5之后出现了调用方法时传递不定参数的情况,如下所示: List<Intrger> list1 = Arrays.asList(new Integer[]{1,2,3}); List<int[]> list2 = Arrays.asList(new int[]{1,2,3}); 在使用不定参数时,可以传入一个数组或是以逗号间隔的参数列表.但是以上两行代码都向方法中传入数组,而传入基本类型数组时并没有按照预想的返回List<Intrger>,而是返回List<i

C语言不定参数

最近,遇到一个c语言的不定参数问题.其实,对于c语言的不定参数问题,只需要三个函数就可以搞定了.这三个函数的头文件是<stdarg.h>,其实下面的三个函数都是一个宏定义(macro).    这三个函数是:    void va_start(va_list ap, last);    type va_arg(va_list ap, type);    void va_end(va_list ap);    如果需要进行其他的一些操作,可以查看一下man手册进行查询.    在这三个函数解释之

Golang的不定参数

在很多语言中都提供了不定参数和函数重载以及函数式语言中得闭包来提高函数的灵活性.如果使用过fmt包里的函数,那么你就已经接触到了Golang的不定参数了.那么如何定义一个自己的不定参数的函数呢? 一.函数定义 首先来看如何定义一个不定参数的函数: func YourFun(v... interface{}){ } 该函数定义,定义了一个接受任何数目任何类型参数的函数.这里特殊的语法是三个点"...",在一个变量后面加上三个点后,表示从该处开始接受不定参数,不同于python,golan

java不定参数String... args

先看两个简单的例子,来感受一下Java的不定长度参数 第一个例子: Java代码   public class VariArgs { public static void main(String[] args) { test(); test("aaa"); test("aaa", "bbb"); test("aaa", "bbb", "ccc"); } public static voi

【转载】c语言中的可变参数编程

在c语言中使用变长参数最常见的就是下面两个函数了: int printf(const char *format, ...); int scanf(const char *format, ...); 那他们是怎样实现支持变成参数的呢?在使用变长参数的函数(这里假设是func)实现部分其实用到了stdarg.h里面的多个宏来访问那些不确定的参数,它们分别是: void va_start(va_list ap, last); type va_arg(va_list ap, type); void va