【C语言】printf函数的简单实现(可变参数列表)

stdarg宏:

可变参数列表是通过宏来实现的,这些宏定义于stdarg.h头文件中,它是标准库的一部分。这个头文件声明一个类型va_list和三个宏va_startva_argva_end。我们可以声明一个类型为va_list的变量,与这几个宏配合使用,访问参数。

 

声明一个va_list类型的变量arg,它用于访问参数列表的未确定部分。这个变量是调用va_start来初始化的。它的第一个参数是va_list的变量名,第2个参数是省略号前最后一个有名字的参数。初始化过程把var_arg变量设置为指向可变参数部分的第一个参数。

 

为了访问参数,需要使用va_arg,这个宏接受两个参数:va_list变量和参数列表中下一个参数的类型。在这个例子中所有的可变参数都是整型。va_arg返回这个参数的值,并使用var_arg指向下一个可变参数。

 

最后,当访问完毕最后一个可变参数之后,我们需要调用va_end

 

可变参数的限制

注意,可变参数必须从头到尾逐个访问。如果你在访问了几个可变参数之后想半途终止,这是可以的,但是,如果你想一开始就访问参数列表中间的参数,那是不行的。

 

1.参数列表中至少有一个命名参数。如果连一个命名参数都没有,就无法使用va_start

2.这些宏是无法直接判断实际存在参数的数量。

3.这些宏无法判断每个参数的是类型。

4.如果在va_arg中指定了错误的类型,那么其后果是不可预测的。

 

我们的my_printf就是用可变参数列表实现的,当然功能比较简单,代码如下:

#include
#include
void my_printf(char *fmt,...)
{
	va_list arg;
	va_start(arg,fmt);
	while(*fmt!='\0')
	{
		switch(*fmt)
		{
		case 'c':
			{
				char c = va_arg(arg,char);
				putchar(c);
				break;
			}
		case 's':
			{
				char *s=va_arg(arg,char*);
				fputs(s,stdout);
				break;
			}
		case 'd':
			{
				int d=va_arg(arg,int);
				char arr[10];
				_itoa(d, arr, 10);
				fputs(arr, stdout);
				break;
			}
	   default:
			{
				//char ch = va_arg(arg, char);
				putchar(*fmt);
				break; 
			}		
		}
		*fmt++;
	}
	va_end(arg);
}
int main()
{
	my_printf("s\nc\nd\n", "bit-tech", 'c',12345);

	return 0;
}


 结果如下所示:

【C语言】printf函数的简单实现(可变参数列表)_第1张图片

 

 

 

你可能感兴趣的:(【C语言】printf函数的简单实现(可变参数列表))