PS:要转载请注明出处,本人版权所有。

PS: 这个只是基于《我自己》的理解,

如果和你的原则及想法相冲突,请谅解,勿喷。

前置说明

  本文作为本人csdn blog的主站的备份。(BlogID=032)

  本文发布于 2017-03-14 17:12:28,现用MarkDown+图床做备份更新。blog原图已丢失,使用csdn所存的图进行更新。(BlogID=032)

环境说明

  无

前言


  在几年前,由于兴趣需要,去研究了c的不定参数问题,当时由于太懒,没有记录任何资料,导致现在实际需要的时候,又要重新来过。呜呼哀哉,太烦了~~~~~~~

c/cpp不定参数


  首先,c不定参数所定义的宏的头文件在 stdarg.h中。在我的系统中在这里:/usr/lib/gcc/x86_64-linux-gnu/5/include/stdarg.h,源码我就不贴了,就简要分析几个重要宏的原理。

//va_list  //此宏依赖与不同的平台和操作系统。
//常用定义:(具体和编译器有关,其实了解透后,都差不多)
typedef char * __builtin_va_list;//(注意可能是这样定义的)
(or)
typedef void * __builtin_va_list;//(注意可能是这样定义的)
typedef __builtin_va_list __gnuc_va_list;
typedef __gnuc_va_list va_list; //_INTSIZEOF(n) //此宏是windows上定义的,目的是为了考虑兼容一些内存地址对齐的系统,n应该占多少字节,如:arm指令集是4字节对齐访问,Thumb指令集是24字节对齐访问.在linux上,gcc编译器会内部定义一个类似的宏,来代表对齐的字节数。其用法是为了根据一个确定的参数,来依次确定不定参数中的每一个参数。
#define _INTSIZEOF(n) \
( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
//一般在一个操作系统中,int类型所占的字节数即为当前对齐的字节数,一般为4或者8。关于此宏的详细分析,以下的分析来至于网上,带入sizeof(n) = 4,8即可明白:
#define _INTSIZEOF(n) ((sizeof(n) + x) & ~(x))
x = sizeof(int) - 1 = 3 = 0000 0000 0000 0011(b)
~x = 1111 1111 1111 1100(b) //va_start(v,l) //此宏获取的是可变参数中的第一个参数的地址
#define __builtin_va_start(v,l) \
( v = (va_list)&l + _INTSIZEOF(l) )//(注意可能是这样定义的),l的地址加上l地址所占的空间
#define va_start(v,l) __builtin_va_start(v,l) //va_arg(v,l)//此宏是使下一个可变参数的地址赋值给v,同时返回开始时v地址所指向的值,注意:此时v已经指向下一个参数地址,但是表达式返回的值是初始v的值,《《这里只需要搞清楚括号优先级就能够明白》》。
#define __builtin_va_arg(v,l) \
( *(l *)((v += _INTSIZEOF(l)) - _INTSIZEOF(l)) )//(注意可能是这样定义的)
#define va_arg(v,l) __builtin_va_arg(v,l) //va_end(v)//此宏处理va_list声明的一个指针,防止出现野指针。有始有终。
#define __builtin_va_end(v)\
( v = (va_list)0 )
#define va_end(v) __builtin_va_end(v)
来个例子(t.c)
#include <stdarg.h>
#include <stdio.h> void ttt(char * fuk,...); int main(int argc, char * argv[]){
ttt("fuk",2333,"this is 23333333333333333");
return 0;
}
void ttt(char * fuk,...){ va_list arg;
int a;
char *b;
va_start(arg,fuk);
a = va_arg(arg,int);
b = va_arg(arg,char *);
va_end(arg); printf("%d,%s\n",a,b);
}

编译

gcc t.c
./a.out

后记


最后的注意:

  1. 变参函数必须有一个以上的指定参数
  2. 变参的《《类型》》必须通过一定方式已知(如格式化字符串,"%d,%s"等等),否则就是定死的
  3. 变参的《《个数》》也必须通过一定方式已知(如格式化字符串,"%d,%s"等等),否则就是定死的
  4. c的函数参数入栈方式是从右到左,栈的地址一般是从高到底。

参考文献


打赏、订阅、收藏、丢香蕉、硬币,请关注公众号(攻城狮的搬砖之路)

PS: 请尊重原创,不喜勿喷。

PS: 要转载请注明出处,本人版权所有。

PS: 有问题请留言,看到后我会第一时间回复。

C 可变参数函数分析(va_start,va_end,va_list...)的更多相关文章

  1. va_start可变参数函数

    void va_start(va_list ap, last); //变参起始地址 type va_arg(va_list ap, type); //下一个参数的地址 void va_end(va_l ...

  2. C语言可变参数函数实现原理

    一.可变参数函数实现原理 C函数调用的栈结构: 可变参数函数的实现与函数调用的栈结构密切相关,正常情况下C的函数参数入栈规则为__stdcall, 它是从右到左的,即函数中的最右边的参数最先入栈. 本 ...

  3. c语言可变参数函数

    c语言支持可变参数函数.这里的可变指,函数的参数个数可变. 其原理是,一般情况下,函数参数传递时,其压栈顺序是从右向左,栈在虚拟内存中的增长方向是从上往下.所以,对于一个函数调用 func(int a ...

  4. C语言学习020:可变参数函数

    顾名思义,可变参数函数就是参数数量可变的函数,即函数的参数数量是不确定的,比如方法getnumbertotal()我们即可以传递一个参数,也可以传递5个.6个参数 #include <stdio ...

  5. C语言中可变参数函数实现原理

    C函数调用的栈结构 可变参数函数的实现与函数调用的栈结构密切相关,正常情况下C的函数参数入栈规则为__stdcall, 它是从右到左的,即函数中的最右边的参数最先入栈.例如,对于函数: void fu ...

  6. C可变参数函数 实现

    转自:http://blog.csdn.net/weiwangchao_/article/details/4857567 C函数要在程序中用到以下这些宏: void va_start( va_list ...

  7. c可变参数函数

    C函数要在程序中用到以下这些宏: <pre lang="c" escaped="true">void va_start( va_list arg_p ...

  8. 【转】C/C++中可变参数函数的实现

    转自:http://www.cnblogs.com/cylee025/archive/2011/05/23/2054792.html 在C语言的stdarg.h头文件中提供了三个函数va_start, ...

  9. C语言可变参数函数的编写

    1. 引言 C语言我们接触的第一个库函数是 printf(“hello,world!”);其参数个数为1个. 然后,我们会接触到诸如: printf(“a=%d,b=%s,c=%c”,a,b,c);此 ...

  10. 可变参数函数(stdarg.h)的使用

    2013/5/3记录: stdarg.h是C语言中C标准函数库的头文件,stdarg是由standard(标准) arguments(参数)简化而来,主要目的为让函数能够接收可变参数.   stdar ...

随机推荐

  1. HarmonyOS 开发入门(一)

    HarmonyOS 开发入门(一) 日常逼逼叨 因为本人之前做过一些Android相关的程序开发,对移动端的开发兴趣比较浓厚,近期也了解到了一些关于华为HarmonyOS 4.0 的事件热点,结合黑马 ...

  2. PVE上启用Intel核显的SR-IOV vGPU

    介绍 Intel SR-IOV vGPU是一种硬件虚拟化技术,它允许多个虚拟机共享单个物理GPU,而不会降低性能.SR-IOV定义了一种标准方法,通过将设备分区为多个虚拟功能来共享物理设备功能.每个虚 ...

  3. Python 元组详细使用

    1. 元组 元组和列表类似,但属于不可变序列,元组一旦创建,用任何方法都不可修改其元素. 元组的定义方式和列表相同,但定义时所有元素是放在一对圆括号"()"中,而不是方括号中. 1 ...

  4. [Ngbatis源码学习] Ngbatis 源码学习之资源加载器 DaoResourceLoader

    Ngbatis 源码学习之资源加载器 DaoResourceLoader DaoResourceLoader 是 Ngbatis 的资源文件加载器,扩展自 MapperResourceLoader.本 ...

  5. CF1853

    你谷的加题速度实在太慢了 被 CF 的题目薄纱 A 可以选任意次 \(i\in [1,n]\),使 \(a[1\sim i]++,a[i+1\sim n]--\).求最少操作次数使得原数列变成非从小到 ...

  6. JavaScript 的灵异事件之一

    场景 在做项目的时候需要用到Ajax 做多次的异步处理数据, 三次Ajax:A --ok--> B --ok--> C 在入参数据相同的情况下,做了两论这个操作,但发现没有发送 A 的 A ...

  7. springboot集成腾讯cos实现文件上传

    腾讯对象存储介绍 对象存储(Cloud Object Storage,COS)是腾讯云提供的一种存储海量文件的分布式存储服务,具有高扩展性.低成本.可靠安全等优点.通过控制台.API.SDK 和工具等 ...

  8. php+bootstrap+jquery+mysql实现购物车项目案例

    获取源码 一键三连后,评论区留下邮箱安排发送:) 介绍 使用php,bootstrap,jquery,mysql实现的简易购物车案例. 通过本案例,你将学习到以下知识点: php 操作 mysql 实 ...

  9. 【树莓派】拷贝系统到新SD卡(系统备份/部署到另一台树莓派上)适用ubuntu 20.04.3

    本教程适用ubuntu 20.04.3 其他版本也大同小异.这种方法能更快的将系统部署下去,如果重新安装一遍加上各种配置相信你会比较疯狂即使做了自动化脚本! 一.树莓派sd卡拷贝 把旧SD卡插入树莓派 ...

  10. String--cannot convert from 'const char *' to 'LPTSTR'错误

    这种错误,很多情况下是类型不匹配 LPTSTR表示为指向常量TCHAR字符串的长指针 TCHAR可以是wchar_t或char,基于项目是多字节还是宽字节版本. 看下面的代码,代码来源:Example ...