C语言函数可变长度参数剖析
C语言中的很多函数的入参被定义为可变参数,最典型的
int printf (const char * fmt, ...)
要对其中的可变参数进行处理,就要用到va_list类型和 VA_START, VA_END, VA_ARG 宏 ,需要包含<stdarg.h>头文件
利用va族函数对不定参数进行解析的过程所示如下:
int my_printf(const char * fmt, ...)
{
va_list struAp;
va_start(struAp, fmt); for (; *fmt; ++fmt)
{
if(*fmt != '%')
{
PUTC(*fmt);
continue;
} fmt++; switch (*fmt)
{
case 'd':
{
int i = va_arg(struAp,int);
PUTC(i);
}
break; default:
break;
}
} va_end(struAp);
}
要了解不定参数的处理方式,就要搞清楚va族函数的实现
#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
/* 实质就是一个char型的指针 */
typedef char * va_list;
/* 将指针偏移一个v的长度,指向后面的地址 */
#define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) )
/* 根据参数类型,取出后面的数据,强制类型转换 */
#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
/* 将指针置为NULL */
#define va_end(ap) ( ap = (va_list)0 )
通过解析fmt字符串,得到后面的参数类型和个数,根据参数类型再加上偏移量就可以找到栈中的不定参数了
函数调用和传参的过程所示如下:
将函数参数与函数调用后下一条指令的地址都压入栈中,然后跳到函数的入口地址。

例如
void func(int param1, double param2,int param3){ }
int main()
{
    func(, 1.2, );
    printf("Over\n"); //设指令地址为0x1234
    return ;
}
执行f(3, 1.2, 4)的函数调用,进入func函数时的堆栈如下:

- 在C语言中,调用一个可变参数函数时,调用者会对每个参数执行“默认实际参数提升(default argument promotions)”
 
——float类型的实际参数将提升到double
  ——char类型的实际参数将提升到int
  ——short类型的实际参数将提升到int
- 在没有函数原型的情况下,char与short类型都将被转换为int类型,float类型将被转换为double类型。
 
——《C语言程序设计》第2版  2.7
类型转换 p36
- 这样写肯定是不对的:
 
c = va_arg(ap,char);
  因为我们无法传递一个char类型参数,如果传递了,它将会被自动转化为int类型。上面的式子应该写成:
  c = va_arg(ap,int);
——《C陷阱与缺陷》p164
C语言函数可变长度参数剖析的更多相关文章
- C语言函数可变参数列表
		
C语言允许使用可变参数列表,我们常用的printf函数即为可变参数函数,C标准库提供了stdarg.h为我们提供了这方面支持:该头文件提供了一些类型和宏来支持可变参数列表,包括类型va_list,宏v ...
 - C语言函数不定参数实现方式
		
函数如何实现不定参数: 由于在C语言中没有函数重载,解决不定数目函数参数问题变得比较麻烦,即使采用C++,如果参数个数不能确定,也很难采用函数重载.对这种情况,提出了指针参数来解决问题. (1)va_ ...
 - GO 函数的参数
		
一.函数 函数的参数 1.1 参数的使用 形式参数:定义函数时,用于接收外部传入的数据,叫做形式参数,简称形参. 实际参数:调用函数时,传给形参的实际的数据,叫做实际参数,简称实参. 函数调用:  ...
 - C 语言函数参数只能传指针,不能传数组
		
今天被要求编写一个C/C++冒泡算法的程序,心想这还不是手到擒来的事儿,虽然最近都是用Javascript程序,很少写C/C++程序,但是好歹也用过那么多年的C语言: 首先想的是怎么让自己的代码看上去 ...
 - Swift 1.1语言函数参数的特殊情况本地参数名外部参数名
		
Swift 1.1语言函数参数的特殊情况本地参数名外部参数名 7.4 函数参数的特殊情况 声明定义有参函数时,为函数的每一个参数都定义了参数名称.根据参数名定义的形式不同,函数参数包括本地参数和外部 ...
 - [转]深度探索C语言函数可变长参数
		
转自:http://www.cnblogs.com/chinazhangjie/archive/2012/08/18/2645475.html 一.基础部分 1.1 什么是可变长参数 可变长参数:顾名 ...
 - C语言函数参数压栈顺序为何是从右到左?(从左向右的话,碰到printf的会陷入死循环)
		
上学期学习了汇编语言,并在操作系统实验中使用了汇编+C语言混合编程,中间也了解了一些C语言与汇编语言的对应关系. 由于汇编语言是底层的编程语言,各种函数参数都要直接控制栈进行存取,在混合编程中,要用汇 ...
 - C语言中函数可变参数解析
		
大多数时候,函数中形式参数的数目通常是确定的,在调用时要依次给出与形式参数对应的所有实际参数.但在某些情况下希望函数的参数个数可以根据需要确定.典型的例子有 大家熟悉的函数printf().scanf ...
 - 【嵌入式开发】C语言 命令行参数 函数指针 gdb调试
		
. 作者 : 万境绝尘 转载请注明出处 : http://blog.csdn.net/shulianghan/article/details/21551397 | http://www.hanshul ...
 
随机推荐
- 【原创】“借贷宝”砸钱,邀请码 GZZKZK2 (注册成功每人可得20现金,可直接提现)。。。而这只是开始
			
作为IT/互联网资深码农的我,从专业技术角度剖析其流程,确认其各个环节控制严格,无欺诈嫌疑, 最佳运气邀请码 : GZZKZK2, 你在注册时值得拥有, 无邀请码无奖励, 亲一定要记住.对 APP操作 ...
 - 修改开机启动等待时间(for Ubuntu12.10)
			
Ubuntu的开机启动等待时间默认是10s,等待时间比较长,每次启动都得按一下回车,于是就想修改一下等待时间.我们可以找到Grub的配置文件(/boot/grub/grub.cfg),在其中进行个性化 ...
 - Asp.NET MVC 拍卖网站,拆解【2】 Asp.NET MVC章回,第(1)节
			
时间和篇幅所限,MVC不会介绍基本的建站过程,请参照博客园技术专题文章传送门 英语足够好的请直接去微asp.net 官网 传送门(强烈推荐,尤其是想使用最新技术的时候更应该直接去官网),本文主要介绍 ...
 - Python的with语句
			
写过多线程程序的人肯定对各种锁很熟悉,尤其是下面这种代码 def lock_usage: lock.Lock() if(...) : lock.Unlock() return lock.Unlock( ...
 - 看看这蛋疼的Java代码
			
项目上要基于现有代码开发,却碰到了很多让人蛋疼的代码.例如下面这个,大家看看能找到多少槽点: public static String addDate(String date, String into ...
 - 浅谈压缩感知(二十八):压缩感知重构算法之广义正交匹配追踪(gOMP)
			
主要内容: gOMP的算法流程 gOMP的MATLAB实现 一维信号的实验与结果 稀疏度K与重构成功概率关系的实验与结果 一.gOMP的算法流程 广义正交匹配追踪(Generalized OMP, g ...
 - [Python爬虫] 在Windows下安装PhantomJS和CasperJS及入门介绍(上)
			
最近在使用Python爬取网页内容时,总是遇到JS临时加载.动态获取网页信息的困难.例如爬取CSDN下载资源评论.搜狐图片中的“原图”等,此时尝试学习Phantomjs和CasperJS来解决这个问题 ...
 - JavaScript简介及基础知识(1)
			
1.JavaScript是什么—它是个脚本语言,需要宿主文件,它的宿主文件是html文件. Javascript是一种脚本语言,比HTML要复杂.不过即便你先前不懂编程,也不用担心,因为Javascr ...
 - mysql忘记密码重置(mac)
			
setp1: 苹果->系统偏好设置->最下边点mysql 在弹出页面中 关闭mysql服务(点击stop mysql server) step2:进入终端输入:cd /usr/local/ ...
 - 开发ERP软件应该遵守的22条规则
			
总结一下做管理软件,有哪些项是经过检验的条款,必须遵守的. 界面篇 1 要保存用户的偏号(profile/favourite). ASP.NET 2.0引入此功能,当用户修改默认的控件的属性时,框架 ...