函数原型: int printf(const char *format[,argument]...)

返 回 值: 成功则返回实际输出的字符数,失败返回-1.

函数说明:

使用过C语言的人所再熟悉不过的printf函数原型,它的参数中就有固定参数format和可变参数(用"…"表示),format后面的参数个数不确定,且类型也不确定,这些参数都存放在栈内。而程序员又可以用各种方式来调用printf,如: 

            printf("%d ",value);   

            printf("%s ",str);   

            printf("the number is %d,string is:%s ",value,str); 

调用printf()函数时,根据format里的格式("%d %f...")依次将栈里参数取出。而取出动作要用到va_arg、va_end、va_start这三个宏定义,再加上va_list。

     (1)va_list事实上是一char *类型,即:

            typedef char* va_list;

     (2)三个宏定义:

        #define _INTSIZEOF(n) ((sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) ) 

为了字节对齐,将n的长度化为int长度的整数倍。

补充:对((sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) ) 的解释

1.举个栗子解释一下内存对齐是什么?

      比方说有一个箱子可以装4个瓶子,我有8个瓶子 ,那么我需要2个箱子如果我有10个瓶

子呢,我不能说我需要10除4,需要2.5个箱子吧。实际上我需要3个箱子,那怎么求我实际需

要的箱子数呢?

用一个容易理解的公式来求上述问题:

设我的瓶子数为B,我需要的箱子数为C,一个箱子最多可以装A个瓶子。

公式:C=(B+A-1)/A

带入几个例子:

B=10,  A=4   得C=13/4 ,C=3        每个箱子最多能装4个瓶子,10个瓶子需要3个箱子

B=14,A=4  得C=17/4  ,C=4        每个箱子最多能装4个瓶子,14个瓶子需要4个箱子

2.细致的分析一下((sizeof(n)+sizeof(int)-1)&~(sizeof(int)-1)) 

((sizeof(n)+sizeof(int)-1)&~(sizeof(int)-1)) :  ((sizeof(n)+sizeof(int)-1)其实就是 (B+A-1)

a、(sizeof(n)是我们需要的实际内存大小,相对于我的瓶子数。

b、sizeof(int)是内存大小分配的最小刻度,相对于箱子最多可以装瓶子的个数。

c、在32位系统中,~(sizeof(int)–1) )展开为~(4-1)=~(00000011b)=11111100b,这样

任何数 & ~(sizeof(int)–1) )后最后两位肯定为0,也就是4的整数倍。

// e、((sizeof(n)+sizeof(int)-1)&~(sizeof(int)-1)) & ~(sizeof(int)-1))其实就是  除以A或者说除

// 以4;((sizeof(n)+sizeof(int)-1)&~(sizeof(int)-1)) 其实就是实现栗子中的(B+A-1)/A。

// (此处描述错误,并不是/4)

总结:

_INTSIZEOF(n)整个做的事情就是将n的长度化为int长度的整数倍。

比如n为5,二进制就是101b,int长度为4,二进制为100b,那么n化为int长度的整数倍就应

该为8。~(sizeof(int) – 1) )就应该为~(4 - 1) = ~(00000011b) = 11111100b,这样任何

数 & ~(sizeof(int) – 1) )后最后两位肯定为0,就肯定是4的整数倍了。

(sizeof(n) + sizeof(int) – 1)就是将大于4m但小于等于4(m + 1)的数提高到大于等于

4(m + 1)但小于4(m + 2),这样再& ~(sizeof(int) – 1))后就正好将原长度补齐到4的倍数

了。

        #define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) ) 

通过该宏定义可以获取到可变参数表的首地址,并将该地址赋给指针ap。

         
 

#define va_arg(ap,type) ( *(type *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) ) 

通过该宏定义可以获取当前ap所指向的可变参数,并将指针ap指向下一个可变参

数。注意,该宏的第二个参数为类型。

        #define va_end(ap)          ( ap = (va_list)0) 

通过该宏定义可以结束可变参数的获取。

可以看出,该函数的参数格式不固定,参数类型不固定。在C语言中使用宏来处理这些可变参数。这些宏看起来很复杂,其实原理挺简单,即根据参数入栈的特点从最靠近第一个可变参数的固定参数开始,依次获取每个可变参数的地址。

 
 

程序员通过这三个宏定义就可以实现对可变参数的处理。例如:

1 #include <stdio.h>  
2   
3 typedef char* va_list;   
4
5 #define _INTSIZEOF(n) ((sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1))   
6 #define va_start(ap,v)  ( ap = (va_list)&v + _INTSIZEOF(v) )   
7 #define va_arg(ap,type) ( *(type *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )   
8 #define va_end(ap)      ( ap = (va_list)0 )   
9   
10 int cal_val(int c, ...)   
11 {   
12         int sum = c;   
13         va_list ap;              //声明指向char型的指针  
14     va_start(ap, c);         //获取可变参数列表的首地址,并赋给指针ap  
15   
16     c = va_arg(ap, int);     //从可变参数列表中获取到第一个参数(返回值即为参数)  
17     while(0 != c)   
18     {   
19         sum += c;   
20         c = va_arg(ap,int);  //循环的从可变参数列表中获取到参数(返回值即为参数)  
21     }  
22     va_end(ap);              //结束从可变参数列表中获取参数  
23     return sum;   
24 }    
25    
26 int main(int argc, char* argv[])   
27 {   
28         int value1, value2; 
29         value1 = cal_val(1,2,3,4,5,6,7,8,9,0);   
30         printf("value1=%d/n",value1);  
31         value2 = cal_val(6,7,8,9,0);   
32

C语言中可变参数的原理——printf()函数的更多相关文章

  1. C语言中可变参数的函数(三个点,“...”)

    C语言中可变参数的函数(三个点,“...”) 本文主要介绍va_start和va_end的使用及原理. 在以前的一篇帖子Format MessageBox 详解中曾使用到va_start和va_end ...

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

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

  3. C语言中可变参数的用法

    原文地址: http://blog.csdn.net/wooin/archive/2006/04/29/697106.aspx   我们在C语言编程中会遇到一些参数个数可变的函数,例如printf() ...

  4. C语言中可变参数的使用

    在C语言程序编写中我们使用最多的函数一定包括printf以及很多类似的变形体.这个函数包含在C库函数中,定义为 int printf( const char* format, ...); 除了一个格式 ...

  5. 可变参数列表与printf()函数的实现

    问题 当我们刚开始学习C语言的时候,就接触到printf()函数,可是当时"道行"不深或许不够细心留意,又或者我们理所当然地认为库函数规定这样就是这样,没有发现这个函数与普通的函数 ...

  6. c 中可变参数的实现

    我们在C语言编程中有时会遇到一些参数个数可变的函数,例如printf()函数,其函数原型为:     例一: int   printf(   const   char*   format,   ... ...

  7. C/C++函数调用时传参过程与可变参数实现原理

    C/C++函数调用时传参过程与可变参数实现原理 C语言的经典swap问题 在学习C语言的时候,我们大都遇到过一些经典例题,这些经典例题背后所代表的是往往是C/C++背后的一些运行原理,比如下面这个示例 ...

  8. [11 Go语言基础-可变参数函数]

    [11 Go语言基础-可变参数函数] 可变参数函数 什么是可变参数函数 可变参数函数是一种参数个数可变的函数. 语法 如果函数最后一个参数被记作 ...T ,这时函数可以接受任意个 T 类型参数作为最 ...

  9. C语言的可变参数在Linux(Ubuntu)与Windows下注意点

    基本上C语言的可变参数原理在不同平台和不同编译器下基本类似(通过函数入栈,从右向左,从高位到低位地址),不过部分实现会有所不同:在使用中需要注意的是: va_list 为char 类型指针,部分调用如 ...

随机推荐

  1. Atom 必装插件

    Atom 必装插件 转载请注明出处. https://blog.csdn.net/Nick_php/article/details/54020956 主题 atom-material-ui 字体配色 ...

  2. 捣鼓FileZilla

    今天突然对ftp服务器感兴趣,于是随意打了一个ftp词条,发现了FZ官网,好奇点进去下载了之后,捣鼓了一会.于是,也写一个小教程记录一下吧,害怕自己以后忘记怎么弄的了. 首先需要用到两个,一个是FZ ...

  3. 解题报告:luogu P1433 吃奶酪

    题目链接:P1433 吃奶酪 我感觉可以改成:[模板]TSP问题(商旅问题) 了. 爆搜\(T\)一个点,考虑状压\(dp\)(还是爆搜). 我们用\(dp[i][j]\)表示现在是\(i\)状态,站 ...

  4. UniGUI 之UniDBGrid(05)

    UniGUI 之UniDBGrid(05) 目录1]DataSource设置2]显示MEMO类型里的文字3]显示悬浮提示4]显示当前记录及总记录数5]读取所有记录,及分页6]在前面加上序号列7]不显示 ...

  5. 如何:使用 as 和 is 运算符安全地进行强制转换(C# 编程指南)

    如何:使用 as 和 is 运算符安全地进行强制转换(C# 编程指南) 由于对象是多态的,因此基类类型的变量可以保存派生类型.若要访问派生类型的方法,需要将值强制转换回该派生类型.不过,在这些情况下, ...

  6. 【剑指Offer面试编程题】题目1504:把数组排成最小的数--九度OJ

    题目描述: 输入一个正整数数组,把数组里所有数字拼接起来排成一个数,打印能拼接出的所有数字中最小的一个.例如输入数组{3,32,321},则打印出这三个数字能排成的最小数字为321323. 输入: 输 ...

  7. Spring和SpringMVC的直接

    1.Spring的常用注解 2.SpringMVC的常用注解

  8. java中的 RSA加密

    package com.cn.test.rsa; import java.math.BigInteger; import java.security.KeyFactory; import java.s ...

  9. Vue二次精度随笔(2)

    1.vue中数组更新是否会引起视图刷新的研究 (1)vue中修改数组可以引起视图刷新的方法 (2)不会引起数组刷新的方法,需要手动进行赋值 (3)有些数组的变化是不能够引起视图的刷新的,一个是修改数组 ...

  10. 在linux环境中如何删除文件

    使用rm -rf 目录名字 命令即可 -r 就是向下递归,不管有多少级目录,一并删除-f 就是直接强行删除,不作任何提示的意思 eg 删除文件夹实例:rm -rf /var/log/httpd/acc ...