C语言实现参数可变函数printf

C语言实现参数可变函数printf

printf函数的功能相信不用多说,这应该是大家写代码用的最多的一个函数了,那它是怎么做到参数可变的呢。首先此功能实现可变参数的关键在于利用stdarg.h头文件中的三个函数

va_start()
va_arg()
va_end()

然后利用putchar函数输出字符到终端,看代码,我尽量把注释写详细。

#include <stdio.h>
#include <stdarg.h>

//函数声明
void printDec(int dec);
void printHex(int hex);
void printOct(unsigned oct);
void printNum(unsigned long num, int base);
void printStr(char *str);
void printFloat(double f);
void printAddr(unsigned long);

void my_Printf(char *fmt, ...)
{
	//可变参数第一步,定义va_list变量
	va_list va_ptr;
	//可变参数第一步,初始化va_ptr,将va_ptr指向第一个可选参数
	va_start(va_ptr,fmt);

	//使用循环,打印完所有格式字符串
	while(*fmt)
	{				
		//普通字符照常输出
		if(*fmt != '%')
		{
			putchar(*fmt++);
			continue;
		}
		
		//处理%后要打印的对应格式
		switch(*(++fmt))//先++是为了跳过%
		{
			case 'd'://打印十进制数
				printDec(va_arg(va_ptr,int));
				break;
			case 'c'://打印字符
				putchar(va_arg(va_ptr,int));
				break;
			case 's'://打印字符串
				printStr(va_arg(va_ptr,char*));
				break;
			case 'x':
			case 'X'://打印十六进制数
				printHex(va_arg(va_ptr,unsigned int));
				break;
			case 'o'://打印八进制数
				printOct(va_arg(va_ptr,unsigned int));
				break;
			case 'f'://打印浮点数
				printFloat(va_arg(va_ptr,double));
				break;
			case 'p'://打印地址
				printAddr(va_arg(va_ptr,unsigned long));
				break;
			case '%'://打印%
				putchar('%');
				break;
			default:
				break;
		}
		fmt++;//别忘记++,继续查询字符
	}
	
	//可变参数最后一步
	va_end(va_ptr);
}

//因为用到此函数打印地址,所以函数的第一个参数用unsigned long最为保险
//16位编译器,地址占16位,2字节,可以使用short或者int保存。
//32位编译器,地址占32位,4字节,可以使用int或long保存
//64位编译器,地址占64位,8字节,可以使用long保存。
void printNum(unsigned long num, int base)
{
	//递归结束条件
	if(num == 0)
	{
		return;
	}
	
	//继续递归
	printNum(num/base,base);
	//逆序打印结果
	putchar("0123456789abcdef"[num%base]);
}

//打印十进制数
void printDec(int dec)
{
	//处理负数的情况
	if(dec < 0)
	{
		putchar('-');
		dec = -dec;
	}
	
	if(dec == 0)
	{
		putchar('0');
		return;
	}
	
	printNum(dec,10);
}

//打印十六进制数
void printHex(int hex)
{
	if(hex == 0)
	{
		putchar('0');
		return;
	}
	
	printNum(hex,16);
}

//打印八进制数
void printOct(unsigned oct)
{
	if(oct == 0)
	{
		putchar('0');
		return;
	}
	
	printNum(oct,8);
}

//打印字符串
void printStr(char *str)
{
	char *p = str;
	while(*p)
	{
		putchar(*p++);
	}
}

//打印浮点数
void printFloat(double f)
{
	int tmp = (int)f;
	//先打印整数部分
	printNum(tmp,10);
	//小数点
	putchar('.');
	
	//再打印小数部分
	f -= tmp;
	if(f == 0)
	{
		int i;
		//浮点型精度至少6位
		for(i = 0; i < 6; i++)
		{
			putchar('0');
		}
		return;
	}
	
	tmp = (int)(f*1000000);
	printNum(tmp,10);
}

void printAddr(unsigned long addr)
{
	printStr("0x");
	
	//内存地址是用十六进制表示的
	printNum(addr,16);
}

int main(int argc, char *argv[])
{
	unsigned long a = 45;
	char ch = 'w';
	char *str = "hello";
	float f = 10.12;
	
	my_Printf("%d %c %s %x %f %o %p %%\n",a,ch,str,567,f,98,&a);
	return 0;
}

此函数实现可能在大厂面试时需手动写出来,所以还是要重视一下的,本人亲身经历,在深信服面试时要求手写此功能代码。

总结:
1.需要记住stdarg.h头文件中的三个函数的用法(三部曲:va_start(),va_arg(),va_end())
2.在while循环中对所以字符进行处理,普通字符照常输出,%后的字符在switch中特殊处理
3.在打印各种进制数据中,利用了递归的思想,这点需要深刻理解一下,其实并不难

备注:此案例中有一个小bug,在打印浮点中printFloat函数中,
tmp = (int)(f*1000000)
这一步改变了小数点后半部分的值(大家可以手动打印一下),导致和原生的printf打印出来的有点出入,但是我在网上暂时没有找到解决方案,有没有大神指点一番~

在这里插入图片描述

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C语言中,我们可以使用标准库中的 `<stdarg.h>` 头文件来定义可变参数函数。下面是一个示例代码: ```c #include <stdio.h> #include <stdarg.h> void printNumbers(int count, ...) { va_list args; // 声明一个va_list类型的变量 va_start(args, count); // 初始化va_list变量 for (int i = 0; i < count; i++) { int num = va_arg(args, int); // 从可变参数列表中获取参数 printf("%d ", num); } va_end(args); // 结束可变参数的获取 printf("\n"); } int main() { printNumbers(5, 1, 2, 3, 4, 5); // 调用可变参数函数 return 0; } ``` 在上述示例代码中,我们首先包含了 `<stdarg.h>` 头文件,然后定义了一个名为 `printNumbers` 的可变参数函数函数的第一个参数 `count` 用于指定可变参数的数量。然后使用 `va_list` 声明了一个可变参数列表的变量 `args`。通过调用 `va_start` 宏,我们对 `args` 进行了初始化,该宏接受两个参数,第一个是 `args`,第二个是最后一个固定参数的前一个参数(即 `count`)。接下来,我们使用 `va_arg` 宏在循环中逐个获取可变参数的值,第一个参数是 `args`,第二个参数是要获取的参数的类型。最后,我们使用 `va_end` 宏结束可变参数的获取。 在 `main` 函数中,我们调用了 `printNumbers` 函数来测试可变参数的使用。 需要注意的是,在C语言中,可变参数函数实现是比较底层和繁琐的,需要开发者手动管理参数列表的获取和结束。因此,在实际开发中,如果可能的话,建议使用C++中提供的更加方便和安全的可变参数函数实现方式。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值