C语言有个很强大的功能,依靠它,实现了printf等这类有着变长参数列表的函数或者宏。它就是在<stdarg.h>里的变长参数。

内容:

va_list   :它是一个适合保存va_start、va_arg和va_end所需要的信息的类型。
va_start :void va_start(va_list ap, parmN); 要在访问所有未命名的参数之前调用宏va_start,宏va_start对ap进行初始化,以便va_arg和va_end使用。
va_arg :type va_arg(va_list, type); 它展开是一个表达式,这个表达式的类型和值跟调用的下一个参数的相同。参数ap应该和va_start初始化的va_list ap相同。
va_end :如果没有执行va_start,或者在返回之前没有调用va_end,那么这种行为未定义。

实现:

还要用到我们在<stddef.h>中提到的<yvals.h>,它里面有两个宏:_AUPBND和_ADNBND

   #define _AUPBND (sizeof (acpi_native_int) - 1) //acpi_native_int 为4字节(32位)(根据机子字数而定)
#define _ADNBND (sizeof (acpi_native_int) - 1)
typedef char* va_list; // 因此,va_list实际上表示的是char*
#define va_start(ap, A) (void)((ap) = (char*)&(A) + _Bnd(A, _AUPBND))
#define va_arg(ap, T) (*(T*) (((ap) += _Bnd(T, _AUPBND)) - _Bnd(T, _ADNBND)))
#define va_end(ap) (void)0 #define _Bnd(X, bnd) (sizeof (X) + (bnd) &~(bnd) )

在实现里我们看到,va_list其实是char*类型,va_start将ap移动到参数列表的首地址,为va_arg的实现做准备,va_end啥都没干。

而va_arg是技巧性最强的一个。首先通过增加va_list对象的内容来使它跳向下一个参数空间的起始位置,然后然退回来指向当前参数的起始位置,然后通过强制类型住哪换把这个指针转换为指定类型的指针,最后解引用获得这个指针以访问存储在数据对象中的值。

在va_start和va_arg里,都用到了一个_Bnd(T, _AUPBND)或_Bnd(T, _ADNBND),由_AUPBND和_ADNBND的宏定义我们知道,它们总是等于在指定的机器中int类型的字节数减1。

事实上, _Bnd宏做的事情就是字节对齐,C语言函数入栈都是字节对齐的,都是所在机器上int类型字节的整数倍进行压栈,这样做可以是内存高效的访问。

sizeof(X)先计算出X类型的大小(假设为s),(bnd) & ~(bnd) 其实bnd就是_AUPBND和_ADNBND(我们假设它们都为3),先将X的大小加3,加了3就会清掉s的二进制数的最低两位,做个假设,假设s为7:00000111,然后加3:00000011,就变为00001010,最后&3的反:(11111100),最后_Bnd计算的结果就是00001000,为8,是int类型大小的整数倍,实现了字节对齐。

《C标准库》——之<stdarg.h>的更多相关文章

  1. 彻底弄清c标准库中string.h里的常用函数用法

    在我们平常写的c/c++程序,一些算法题中,我们常常会用到c标准库中string.h文件中的函数,这些函数主要用于处理内存,字符串相关操作,是很有用的工具函数.而且有些时候,在笔试或面试中也会出现让你 ...

  2. 走进C标准库(8)——"string.h"中函数的实现相关字符串操作函数

    我的strcat: char *strcat(char *dest,char *src) { char * reval = dest; while(*dest) dest++; while(*src) ...

  3. 走进C标准库(3)——"stdio.h"中的getc和ungetc

    接前文. 再来看看getc和ungetc的实现.在看这两个函数的实现之前,我们先来想一想这两个函数分别需要做的工作. int getc(FILE *stream) 说明:函数getc从stream指向 ...

  4. 走进C标准库(2)——"stdio.h"中的fopen函数

    其他的库文件看起来没有什么实现层面的知识可以探究的,所以,直接来看stdio.h. 1.茶余饭后的杂谈,有趣的历史 在过去的几十年中,独立于设备的输入输出模型得到了飞速的发展,标准C从这个改善的模型中 ...

  5. 走进C标准库(1)——assert.h,ctype.h

    默默觉得原来的阅读笔记的名字太土了,改了个名字,叫做走进C标准库. 自己就是菜鸟一只,第一次具体看C标准库,文章参杂了对<the standard C library>的阅读和对源码的一些 ...

  6. C 非标准库(conio.h)

    所谓的 C 标准库(C standard library),是指在 ISO C 或者 POSIX 标准中定义的: POSIX is a superset(超集) of the standard C l ...

  7. 走进C标准库(4)——"stdio.h"中的putc

    花了点时间把园子弄得好看了点,现在继续. 函数名: putc 功  能: 输出一字符到指定流中 用  法: int putc(int ch, FILE *stream); #define _putc_ ...

  8. 走进C标准库(5)——"stdio.h"中的其他部分函数

    函数介绍来自:http://ganquan.info/standard-c/ 函数名: freopen 功  能: 替换一个流 用  法: FILE *freopen(char *filename, ...

  9. 走进C标准库(6)——"string.h"中函数的实现memchr

    我写的memchr: void *memchr(const void *buf, char ch, unsigned count){ unsigned ; while(*(buf++) != ch & ...

  10. 走进C标准库(7)——"string.h"中函数的实现memcmp,memcpy,memmove,memset

    我的memcmp: int memcmp(void *buf1, void *buf2, unsigned int count){ int reval; while(count && ...

随机推荐

  1. Linux - gcc和g++的区别

    一般linux系统都自带了gcc编译器的,你可以用你的安装光盘去安装,如果你是觉得自带的gcc版本太低了,可以去gcc的官方网站可以下载到,编译需要很长的时间,如果你只编译C或者C++可以只下载gcc ...

  2. 提升web响应速度的思路

    web响应(主要指加载网页类,不包括大文件下载,看视频)的核心瓶颈在于延迟,不在于带宽. 从感性认知的角度,由于存在tcp的慢启动,所以往往速率还未达到带宽值时,访问就已经结束:另外,没有交互就没有延 ...

  3. Spring学习笔记之整合struts

    1.现有项目是通过 <action    path="/aaaaAction"                type="org.springframework.w ...

  4. 创建条形码图像易用的控制字符编码功能的条形码控件Native Crystal Reports Barcode Generator

    Native Crystal Reports Barcode Generator是一个对象,它可以很容易地被嵌入到一个Crystal Report中用于创建条形码图像.一旦此条形码被安装在一个报表中, ...

  5. 设置隐藏activity的标题栏

    values->styles.xml中设置:stylename="AppTheme"parent="Theme.AppCompat.Light.NoActionBa ...

  6. Oracle 中批量导入大量测试数据的方法

    执行如下批处理命令: declare  maxrecords constant int := 100000;  i int := 1;begin  for i in 1 .. 20000 loop   ...

  7. 2016 - 1- 22 NSURLConnetction --- GET请求

    ---恢复内容开始--- 一: 给服务器发送一个简单的GET请求 1.同步 // 发送一个GET请求给服务器 // 0.请求路径 NSURL *url = [NSURL URLWithString:@ ...

  8. linux基础命令学习五(软件包管理、下载管理)

    Linux 软件包管理   本文主要是记录下RedHat系列的软件包管理. 内容分为以下二个部分:二进制包的管理,源代码包的管理 一.二进制包的管理 1.1概念 主要有RPM和YUM这两种包管理. 两 ...

  9. Android.mk中添加宏定义

    在Boardconfig.mk 中添加一个 IS_FLAG := true 由于Boardconfig.mk和各目录的Android.mk是相互关联的 所以我们可以在Android.mk 中添加 一个 ...

  10. LeetCode----Word Ladder 2

    Given two words (start and end), and a dictionary, find all shortest transformation sequence(s) from ...