C语言进阶45:函数参数的秘密《下》——调用约定和可变参数程序

函数参数的计算次序是依赖于编译器实现的,那么函数参数的入栈次序是如何确定的呢?

strcpy(s, "willwilling");

哪个参数先入栈呢?——s先入栈。

调用约定:
参数会传递给被调用的函数

返回值会被返回给函数调用者

调用约定描述参数如何传递到栈中以及栈的维护方式

参数传递顺序
调用栈清理

调用约定是编译器的一部分,且各个编译器的调用约定并不相同。

调用约定是预定义的,可理解为调用协议

调用阅读通常用于库调用和库开发的时候:

从右到左依次入栈:
_stdcall, _cdecl, _thiscall
从左到右依次入栈
_pascal, _fastcall

当一个C语言程序调用了一个别的编译器(比如pascal)编译的库,那么如果不显式声明调用约定,则很有可能出现错误。
调用约定规定了函数参数的入栈顺序
当主程序使用第三方库的时候,需要考虑调用约定是否一致

问题提出如果编写一个计算n个数平均值的函数?

                 不使用数组来完成程序编写。

可变参数程序。

C语言可以定义参数可变的函数
参数可变函数的实现依赖于 stdarg.h 头文件
va_list 参数集合
va_arg 取具体的参数值
va_start 标识参数访问的开始
va_end 标识参数访问的结束
#include <stdio.h>
#include <stdarg.h>

float average(int n, ...)
{
	va_list args;  //先声明变量
	int i = 0;
	float sum = 0;
	
	va_start(args, n); //调用这个变量
	
	for(i=0; i<n; i++)
	{
		sum += va_arg(args, int);  //取变量的值,定义数据类型
	}
	
	va_end(args);    //结束访问
	
	return sum / n;
	
}

int main()
{
	printf("%f\n", average(5,1,2,3,4,5));  //对于printf()函数,  %用来标识参数数量
	printf("%f\n", average(4,1,2,3,4,5));   //“5”这个参数并不会被访问
	
	return 0;
}
可变参数的限制
可变参数必须从头到尾按照顺序逐个访问
参数列表中至少要存在一个 确定的命名参数
可变参数函数无法确定实际存在的参数的 数量

可变参数函数无法确定参数的实际类型

注意:va_arg中如果指定了错误的类型,那么结果是不可预测的。

小结:
调用约定指定了函数参数的入栈顺序以及栈的清理方式
可变参数是C语言提供的一种函数设计技巧
可变参数的函数提供了一种很方便的函数调用方式
可变参数必须顺序的访问,无法直接访问中间的参数值


猜你喜欢

转载自blog.csdn.net/qq_28388835/article/details/80381229