模拟实现printf函数

本文探讨了C语言中的可变参数概念,通过分析printf函数的模拟实现,阐述了如何使用va_list、va_start、va_arg和va_end等宏来处理可变参数列表。文中指出,可变参数必须按顺序访问,且需要至少一个命名参数,同时强调了在使用可变参数时需要注意的类型匹配问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

可变参数

所谓可变参数,就是可以用函数传一个、两个或者多个不同数量(不固定)参数的函数。
比如

printf("hello");//一个参数
printf("%d",10);//两个参数
printf("%d,%d,%s",10,20,"hello");//四个参数

printf函数就是典型的可变参数的函数。

用一段代码简单了解一下可变参数列表。

#include <stdio.h>
#include <stdarg.h>
int average(int n, ...)
{
    va_list arg;
    int i = 0;
    int sum = 0;
    va_start(arg, n);
    for(i=0; i<n; i++)
    {
        sum += va_arg(arg, int);
    }
    return sum/n;
    va_end(arg);
}
int main()
{
    int a = average(2, 1, 3);
    int b = average(3, 1, 2, 3);
    printf("a = %d\n", a);
    printf("b = %d\n", b);
    return 0;
}

这段代码用来求不同参数个数的数的平均值。
可变参数列表中主要用的主函数有

va_list arg;
va_start(arg,n);
va_arg(arg,int);
va_end(arg);

其实这几个代码都是宏,宏可以传类型,函数不可以。
将他们转到定义可以看到对应的代码。
这里写图片描述
可以看到va_list是类型重命名。将arg定义为char *类型的指针。
这里写图片描述
将va_start转到定义,发现这又是一个宏,再将其转到定义。
这里写图片描述

可变参数列表主要使用到的函数:
  • _crt_va_start(ap,v)
    这是用来初始化的,将arg变量直接在内存中指向可变参数部分的第一个参数(函数堆栈)。函数堆栈中地址都是由低到高排列,然后传参都是从右往左传。所以v的地址下面压的就是第一个未知参数的地址。
  • _crt_va_arg(ap,t)
    接收可变参数列表下一个参数的类型,将其强制转换成对应的类型再解引用访问到未知参数。将ap自增指向下一个可变参数,然后再将表达式值减回来指向本次可变参数。
  • _crt_va_end(ap)
    将0强制转换成指针也就是空指针赋给ap。初始化指针ap。
  • _INTSIZEOF(n)
    如果n是1-4的数,返回4,如果是5-8将返回8.这是一个将n向上取整的代码(向上取4整)。
    因为宏在编译期间会被替换到函数内部,所以我们可以将宏直接替换到函数内部。
    将宏替换到函数内部:
#include <stdio.h>
#include <stdarg.h>
int average(int n, ...)
{
    //va_list arg;
    char * arg;
    int i = 0;
    int sum = 0;
    //va_start(arg, n);
    arg = (char*)&n + 4;
    for(i=0; i<n; i++)
    {
        //sum += va_arg(arg, int);
        sum = sum + (*(int *)((arg+=4)-4));
    }
    return sum/n;
    //va_end(arg);
    arg = (char *)0;
}
int main()
{
    int a = average(2, 1, 3);
    int b = average(3, 1, 2, 3);
    printf("a = %d\n", a);
    printf("b = %d\n", b);
    return 0;
}

可变参数要注意以下几个地方:

  • 可变参数必须从头到尾逐个访问,如果在访问了几个可变参数之后想半途终止是可以的,但是如果一开始就想访问中间或者后面的参数,是不可行的。
  • 可变参数列表中至少有一个命名参数,否则将无法使用va_start
  • 可变参数列表中用到的宏不能判断所传参数的数量
  • 可变参数列表中用到的宏不能判断参数类型,需要使用者自己输入,如果制定了错误的参数类型,那么将一步错步步错。所以一定要谨慎。
下面再简单的模仿一下printf函数:
#include <stdio.h>
#include <stdarg.h>
#include <assert.h>
#include <string.h>
#include <math.h>

void my_printf(const char *ret)
{
    assert(ret);
    while(*ret)//逐步输出每一个字符
    {
        putchar(*ret);
        ret++;
    }
}

int my_print(int i)
{
    int j = 0;
    int n = 0;
    int count = 0;
    n=i;
    while(n)//计算数字位数
    {
        n=n/10;
        j++;
        count++;
    }
    while(j>=1)//逐步输出每一位
    {
        int k = (int)(i/(pow(10,(j-1))));
        putchar(k+'0');
        j--;
        i = i%10;
    }
    return count;
}

int print(const char *format,...)
{
    int count = 0;//因为printf的返回值是输出的字符个数
                  //所以我们定义一个计数器
    const char *start = format;//定义一个指针来接收这个字符串
    va_list arg_list;
    va_start(arg_list,format);
    assert(format);
    while(*start)
    {
        if(*start == '%')//如果碰到%号则下一个将进行输出
        {
            start++;//加加一次跳到下个字符准备执行下一条输出
            switch(*start)//根绝字符选择对应输出
            {
            case 's'://输出字符串
                {
                    char *ret = 
                    va_arg(arg_list,char *);
                    my_printf(ret);//打印字符串
                    count = count + strlen(ret);//计数器增加字符串长度大小
                }
                break;
            case 'c'://输出字符
                {
                    char tmp = va_arg(arg_list,char);
                    putchar(tmp);//直接打印字符
                    count++;//计数器加1
                }
                break;
            case 'd'://输出数字
                {
                    int i = va_arg(arg_list,int);
                    count = count + my_print(i);//输出数字,
                                //并且计数器增加数字位数
                }
                break;
            default:
                {
                    putchar(*start);
                    count++;
                }
                break;
            }
        }
        else
        {
            putchar(*start);
            count++;
        }
        start++;
    }
    va_end(arg_list);
    return count;
}

int main()
{
    print("hello=%s,%c,%c,%c,%d","hello"
            ,'w','o','r','d'
            ,100);
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值