C标准库之stdarg

✏ 1、可变长参数

C语言支持可变长参数,正常情况下C的函数参数入栈规则为__stdcall,它是从右到左的,即函数中的最右边的参数最先入栈。例如,对于函数:

void func(int a, char b, int c, double d, int e) {
   
    int f = 0;
    printf("&a = 0x%p\n", &a);
    printf("&b = 0x%p\n", &b);
    printf("&c = 0x%p\n", &c);
    printf("&d = 0x%p\n", &d);
    printf("&e = 0x%p\n", &e);
    printf("&f = 0x%p\n", &f);
}
// 输出
&a = 0x0058F968   
&b = 0x0058F96C  // 4  字对齐
&c = 0x0058F970  // 4
&d = 0x0058F974  // 4
&e = 0x0058F97C  // 8
&f = 0x0058F954  

用户栈是从上往下生长的,先占用高地址的空间,再占用低地址空间。对于在32位系统的多数编译器,每个栈单元的大小都是sizeof(int),而函数的每个参数都至少要占一个栈单元大小。对于固定参数列表的函数,每个参数的名称、类型都是直接可见的,他们的地址也都是可以直接得到的。

按照C标准的说明,支持变长参数的函数在原型声明中,必须有至少一个最左固定参数(这一点与传统C有区别,传统C允许不带任何固定参数的纯变长参数函数),这样我们可以得到其中固定参数的地址,根据上面的参数入栈顺序,我们可尝试写一个可变长参数的函数:

void var_args_func(const char* fmt, ...)
{
   
    char* ap;
    ap = ((char*)&fmt) + sizeof(fmt);
    printf("%d\n", *(int*)ap);
    ap = ap + sizeof(int);
    printf("%d\n", *(int*)ap);
    ap = ap + sizeof(int);
    printf("%s\n", *((char**)ap));
}

int main()
{
   
    var_args_func("%d %d %s\n", 4, 5, "hello world");
    return 0;
}

解释:用ap获取第一个变参的地址,我们知道第一个变参是4,一个int 型,所以我们用(int)ap以告诉编译器,以ap为首地址的那块内存我们要将之视为一个整型来使用,(int)ap获得该参数的值;接下来的变参是5,又一个int型,其地址是ap + sizeof(第一个变参),也就是ap + sizeof(int),同样我们使用(int)ap获得该参数的值;最后的一个参数是一个字符串,也就是char,与前两个int型参数不同的是,经过ap + sizeof(int)后,

  • 9
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
引用\[1\]:在你提供的代码中,可以看到使用了标准库函数printf来实现串口打印。在main函数中,通过调用printf函数来输出"ok"。\[1\] 引用\[3\]:为了实现串口打印标准库,你需要在usrat.c文件中添加一些代码。具体来说,你需要包含头文件stdarg.h、string.h和stdio.h,并定义一个名为UsartPrintf的函数。这个函数使用了可变参数列表,可以接受不定数量的参数。在函数内部,使用vsnprintf函数将格式化后的字符串存储在UsartPrintfBuf数组中,然后通过调用HAL_UART_Transmit函数将字符串逐个发送到USARTx串口。\[3\] 所以,通过在代码中使用标准库函数printf和添加UsartPrintf函数,你可以实现stm32的串口打印标准库。 #### 引用[.reference_title] - *1* [STM32F4xx系列标准库函数之串口初始化和打印](https://blog.csdn.net/longjintao1/article/details/124944887)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [关于STM32系列串口打印乱码原因解析](https://blog.csdn.net/newzhpfree/article/details/125405898)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [STM32系列(HAL库) ——使用串口打印的3种方式](https://blog.csdn.net/lwb450921/article/details/127339596)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值