C语言中的可变参数函数的浅析(以Arm 程序中的printf()函数实现为例) .
我们在C语言编程中会遇到一些参数个数可变的函数,一般人对它的实现不理解。例如Printf():
Printf()函数是C语言中非常常用的一个典型的变参数函数,它
的原型为: int printf( const char* format, ...);
它除了一个参数format固定外,后面的参数的个数和类型是不确定的,如下列三种调用方法:
1. printf(“%d\n”, i);
2. printf(“%s\n”, “Hello World”);
3. printf(“The result is %d, name is %s”, i, “Lily”);
使用可变参数时,需要用到的库函数:va_list 、va_start、va_arg、va_end,要包含头文件<stdarg.h>
使用可变参数的步骤:
1)首先在函数里定义一个va_list型的变量,如arg
2)然后用va_start宏初始化变量arg,这个宏的第二个参数是第一个可变参数的前一个参数,是一个固定的参数.
3)然后用va_arg返回可变的参数.依次取可变参数,va_arg的第二个参数是你要返回的参数的类型
4)最后用va_end宏结束可变参数的获取.
现在你就可以在函数里使用第二个参数了.如果函数有多个可变参数的,依次调用va_arg获取各个参数.
应用:在ARM调用串口打印时,是没有Printf()的,这时就需要自己写一个打印函数(前提是已经实现串口的字符、字符串及整数打印)
void Uart_Printf(const char *format, ...)
{
va_list unnamed_p;
char *p, *sval;
unsigned int value_i;
/* unnamed_p point to first unnamed argument */
va_start( unnamed_p, format);
for ( p=(char *)format; *p!='\0'; p++ )
{
if ( *p != '%' )//循环检测format的每位字符,没有遇到“%”,//则输出该字符
{
Uart_Put_Char( *p );
continue;
}
//如果遇到“%”,则是格式控制符,向下运行,指针p加1,确定是//“%d”、“%s”等,然后相应处理
switch ( *++p )
{
case 'd'://输出十进制整数
value_i = va_arg( unnamed_p, unsigned int );
Out_Put_Int( value_i, 10 );
break;
#if 0
case 'c'://输出字符
{
sval = va_arg(unnamed_p, char*);
Uart_Put_Char(*sval);//串口输出字符函数
break;
}
#endif
case 's'://输出字符串
for (sval = va_arg(unnamed_p, char*); *sval; sval++)
Uart_Put_Char( *sval ); //此为串口输出字符函数
break;
case 'x': //X表示16进制
value_i = va_arg( unnamed_p, unsigned int );
Out_Put_Int( value_i, 16 );//此为串口输出无符号整数函数,//第一个参数为要输出的整数,第二个参数为进制,表示以多少进制//的格式输出
break;
default: //输出字符
Uart_Put_Char( *p );
break;
}
}
va_end( unnamed_p ); //结束可变参数的获取
}
C语言中的可变参数函数的浅析(以Arm 程序中的printf()函数实现为例) .的更多相关文章
- C语言中的可变参数函数
C语言编程中有时会遇到一些参数个数可变的函数,例如printf()函数,其函数原型为: int printf( const char* format, ...); 它除了有一个参数format固定以外 ...
- 在C/C++函数中使用可变参数
原文链接地址:http://blog.csdn.net/djinglan/article/details/8425768 下面介绍在C/C++里面使用的可变参数函数. 先说明可变参数是什么,先回顾一下 ...
- C语言中的可变参数-printf的实现原理
C语言中的可变参数-printf的实现原理 在C/C++中,对函数参数的扫描是从后向前的.C/C++的函数参数是通过压入堆栈的方式来给函数传参数的(堆栈是一种先进后出的数据结构),最先压入的参数最后出 ...
- C函数和宏中的可变参数
一:调用惯例 函数的调用方和被调用方对函数如何调用应该有统一的理解,否则函数就无法正确调用.比如foo(int n, int m),调用方如果认为压栈顺序是m,n,而foo认为压栈顺序是n, m,那么 ...
- C/C++函数中使用可变参数
先说明可变参数是什么,先回顾一下C++里面的函数重载,如果重复给出如下声明: int func(); int func(int); int func(float); int func(int, int ...
- 【转】C,C++中使用可变参数
可变参数即表示参数个数可以变化,可多可少,也表示参数的类型也可以变化,可以是 int,double还可以是char*,类,结构体等等.可变参数是实现printf(),sprintf()等函数的关键之处 ...
- [C++]C,C++中使用可变参数
可变参数即表示参数个数可以变化,可多可少,也表示参数的类型也可以变化,可以是int,double还可以是char*,类,结构体等等.可变参数是实现printf(),sprintf()等函数的关键之处, ...
- C# 中的可变参数方法(VarArgs)
首先需要明确一点:这里提到的可变参数方法,指的是具有 CallingConventions.VarArgs 调用约定的方法,而不是包含 params 参数的方法.可以通过MethodBase.Call ...
- Java中的可变参数以及foreach语句
Java中的可变参数的定义格式如下: 返回值类型 方法名称(类型 ... 参数名称){} foreach语句的格式如下: for ( 数据类型 变量名称 :数据名称){ ... } public ...
- 关于Retrofit网络请求URL中含有可变参数的处理
开题:在此默认各位看官对Retrofit.以及Okhttp已经有过一定的了解及应用,所以今天我们不谈基础入门的东西,今天我们谈在Retrofit请求接口管理类中URL参数含有动态参数的处理方式.一般我 ...
随机推荐
- PyCharm运行Nosetests并导出测试报告
1. Pycharm运行Nosetests PyCharm可以使用两种方法,运行Nosetests测试文件: 1) 图形用户界面GUI a) 在PyCharm中,选中测试文件,如Tests/test_ ...
- Sonar项目主要指标以及代码坏味道详解
更多原创测试技术文章同步更新到微信公众号 :三国测,敬请扫码关注个人的微信号,感谢! 原文链接:http://www.cnblogs.com/zishi/p/6766994.html 众所周知Sona ...
- win7 重装 docker 启动后无法启动错误解决
描述 win7 重新安装Docker 后启动 Docker Quickstart Terminal 出现如下错误 Starting "default"... (default) ...
- DG环境的日常巡检
DG环境的日常巡检 目录 1.DG环境的日常巡检4 1.1.主库环境检查4 1.1.1.主库实例启动状态检查4 1.1.2.主库启动模式检查4 1.1.3.主库DG环境的保护模式检查4 1.1.4.主 ...
- 自兴人工智能------Python语言的变量认识及操作
今天我给大家介绍的是python中的Number变量,与c++,java有些不同,下面让来为大家介绍: 在python中是不用声明变量类型的,不过在使用变量前需要对其赋值,没有值得变量是没有意义的,编 ...
- BZOJ 3720: Gty的妹子树 [树上size分块]
传送门 题意: 一棵树,询问子树中权值大于$k$的节点个数,修改点权值,插入新点:强制在线 一开始以为询问多少种不同的权值,那道CF的强制在线带修改版,直接吓哭 然后发现看错了这不一道树上分块水题.. ...
- POJ 3537 Crosses and Crosses [Multi-SG]
传送门 我也不知道为什么枚举vis必须加上一个边界才能A 以后还是都加上吧 #include <iostream> #include <cstdio> #include < ...
- 运行自己的 DaemonSet - 每天5分钟玩转 Docker 容器技术(131)
本节以 Prometheus Node Exporter 为例演示如何运行自己的 DaemonSet. Prometheus 是流行的系统监控方案,Node Exporter 是 Prometheus ...
- I can do it
If it doesn't challange you ,it won't change you.
- nginx的location优先级
在nginx配置文件中,location主要有这几种形式: 1. 正则匹配 location ~ /abc { } 2. 不区分大小写的正则匹配 location ~* /abc { } 3. 匹配路 ...