參考文档: http://bbs.csdn.net/topics/70288067

Owed by: 春夜喜雨 http://blog.csdn.net/chunyexiyu 转载请标明来源

本文的二个重点:

1. 可变參数实际上通过首个參数的地址来获取其他參数的地址。由于是顺序存储传过来的

1.      可变參数为了处理方便。全部的浮点型都是依照double型压栈。

因此,像printf採用的可变參数,它直接使用%f显示double型。不须要管float型。

关注printf的精度,须要先关注printf的实现,关于printf的实现,我们就要关注一下可变參数的实现:

变參的一个例了:

void
testIntAndFloat(int
type, ...)

{

va_list argptr;

va_start(argptr,
type);

if (type == 0 )

{

int n = va_arg(
argptr, int );

printf( "%d\n",
n );

}

else if (type == 1)

{

double d =
va_arg( argptr,
double); // 必须使用double接收,后边会说明原因

printf( "%f\n",
d);

}

else

{}

}

測试:

float
f1 = 12345;

testIntAndFloat(0,
);

testIntAndFloat(1,
f1, f1);

输出:

123

12345.0000

查看变參相关的定义:

typedef
char *  va_list;

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

#define
_ADDRESSOF(v)   ( &reinterpret_cast<constchar &>(v))

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

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

//_ADDRESSOF(v):获取v的地址,并转为char*型

//_INTSIZEOF(n):实现4字节对齐

//va_start (ap,v):
获取參数v之后的地址

//va_arg(av, t): ap地址向后移t类型size,并返回ap地址之前t类型变量的值

看一个输出%d的样例:

__int64
i = 123456;

printf("%d,%d \n",i,
i);

结果为: 123456, 0

printf("%d,%d,%d,%d \n",i,
i);

结果为: 123456,0,123456,0

printf("%lld,%lld \n",i,
i);

结果为: 123456, 123456

从上面的内容能够看出来:

%d时,每次读取的位数是4位,而int64有8位。所以假设使用%d的话,分两次读取完。

(因为是little endian。先读低位,再读高位,先读到123456。再读到0)

再看输出%f的样例

float
f = 123456789123456789.0;

double fd = f;

double
d = 123456789123456789.0;

printf("%f\%f\n%f\n",f,fd,
d);

结果为:

1234567939550609400.0000

1234567939550609400.0000

1234567890123456800.0000

能够看出:

float实际是被转成double型存储显示的

(float的精底一般是6-7位。double是15-16位,printf的时候,显示的位数是按double算的。)

再看这个样例,执行时观察内存信息:

(观察内存方式: VC->调试->窗体->内存)

void
testArgs(int
type, ...)

{

va_list argptr;

va_start(argptr,
type);

for (int i=0;
i<8; i++)

{

int* pNumber = (int*)argptr;

float* pFloat = (float*)argptr;

va_arg(argptr,
int);

printf(("\naddrss:%d int value: %d  float value: %f"),
pNumber, *pNumber, *pFloat);

}

}

void
TestFuncArgs(int
n1 = 31, float
f1=31.0, double
d1=31.0, char
c1=31, bool b1=31,
short s1=31,
__int64 n2=31,
int n3=31)

{

printf("\nf1=%f",
f1);

printf(("\naddrss of arg(int): %x, value: %d"), &n1,
n1);

printf(("\naddrss of arg(float): %x, value: %d|%d, float:%f"), &f1,
f1, f1);

printf(("\naddrss of arg(double): %x, value: %d|%d, float:%f"), &d1,
d1, d1);

printf(("\naddrss of arg(char): %x, value: %d"), &c1,
c1);

printf(("\naddrss of arg(bool): %x, value: %d"), &b1,
b1);

printf(("\naddrss of arg(short): %x, value: %d"), &s1,
s1);

printf(("\naddrss of arg(__int64): %x, value: %d"), &n2,
n2);

printf(("\naddrss of arg(int): %x, value: %d"), &n3,
n3);

testArgs(n1,
f1, d1, c1,
b1, s1,
n2, n3);

}

当按函数定參參数形式传递TestFuncArgs时:

Int/char/bool/short占用4字节

Float占用4字节

Double点用8字节

__int64占用8字节

内存中连续显示

1f 00 00 00 --int

00 00 f8 41 --float

00 00 00 00 00 00 3f 40 --double

1f 00 00 00 --char

01 00 00 00 --bool

1f 00 00 00 --short

1f 00 00 00 00 00 00 00
–int64

1f 00 00 00 00 00 -- int

当採用不定參数传递testArgs时:

其他均不变,但float为8字节存储,这个是须要很注意的一个事情。

使用%d打印时,仅仅会取4字节,须要使用两个%d%d才干打印一个float。

内存中连续显示

1f00 00 00 -- int

00 00 00 00 00 00 3f 40 --float

00 00 00 00 00 00 3f 40 --double

1f 00 00 00 --char

01
00 00 00 --bool

1f 00 00 00 --short

1f 00 00 00 00 00 00 00
–int64

1f 00 00 00--double 
 
Owed by: 春夜喜雨 http://blog.csdn.net/chunyexiyu 转载请标明来源
 

Printf可变參数使用的更多相关文章

  1. C语言利用va_list、va_start、va_end、va_arg宏定义可变參数的函数

    在定义可变參数的函数之前,先来理解一下函数參数的传递原理: 1.函数參数是以栈这样的数据结构来存取的,在函数參数列表中,从右至左依次入栈. 2.參数的内存存放格式:參数的内存地址存放在内存的堆栈段中, ...

  2. java课堂练习之可变參数与卫条件

    /*  有人邀请A,B,C,D,E,F 6个人參加一项会议,这6个人有些奇怪.由于他们有非常多要求,已知:  1)A,B两人至少有1人參加会议:  2)A,E,F 3人中有2人參加会议.  3)B和C ...

  3. OC可变參数的函数实现va_start、va_end、va_list的使用

    一.简单介绍 我们常常在编程的时候看见类似这种代码,如图1.1 图1.1 或者是这种可变參数,如图1.2 图1.2 二.基本知识介绍 在学习怎样写这样的格式的函数前,先简介几个经常使用的宏: 下面摘自 ...

  4. iOS 处理方法中的可变參数

    ## iOS 处理方法中的可变參数 近期写了一个自己定义的对话框的demo,想模仿系统的UIAlertView的实现方式.对处理可变參数的时候,遇到了小问题,于是谷歌了一下.写下了处理问题的方法.记录 ...

  5. Effective JavaScript Item 21 使用apply方法调用函数以传入可变參数列表

    本系列作为Effective JavaScript的读书笔记. 以下是一个拥有可变參数列表的方法的典型样例: average(1, 2, 3); // 2 average(1); // 1 avera ...

  6. java 可变參数

    我们在某些特定的需求环境下,可能要对某一个方法中的參数进行一些操作,并且这些方法中的參数是不规定的,那么问题来了,我们该怎么办呢? java事实上就为我们考虑了这样的情况,那就是使用可变參数 可变參数 ...

  7. java 可变參数列表

    Java SE5加入了可变參数列表特性 參数能够这样定义.(Object-args).可变參数用"..."来定义,args是可变參数的数组.举个样例: package sample ...

  8. java之 ------ 可变參数和卫条件

    可变參数:适用于參数个数不确定.类型确定的情况,java把可变參数当做数组处理. 可变參数必须位于最后一项.当可变參数个数多于一个时,必将有一个不是最后一项,所以仅仅支持有一个可变參数. 可变參数的书 ...

  9. 增强for循环、Map接口遍历、可变參数方法

    增强for循环 1.for循环能做得事情.增强for循环大部分都能做(假设要想获得下标的时候就必须使用简单for循环了) 2.增强for有时候可以方便的处理集合遍历的问题,可是集合的标准遍历是使用迭代 ...

随机推荐

  1. CSAPP学习笔记—虚拟内存

    CSAPP学习笔记—虚拟内存 符号说明 虚拟内存地址寻址 图9-12展示了MMU如何利用页表来实现这种映射.CPU中的一个控制寄存器,页表基址寄存器(Page Table Base Register, ...

  2. document.execCommand

    document.execCommand 在firefox浏览器执行不好,但是在其他浏览器有时候使用会非常方便. 比如在input标签中使用: onkeyup="if(isNaN(value ...

  3. 如何部署 sources and javadoc jars

    mvn org.apache.maven.plugins:maven-deploy-plugin:2.8.2:deploy-file -Durl=file:///home/me/m2-repo \ - ...

  4. 通过 Terracotta实现基于Tomcat的Web应用集群

    [转]通过 Terracotta实现基于Tomcat的Web应用集群 博客分类: 企业应用面临的问题 Java&Socket 开源组件的应用 tomcatweb session集群服务器负载均 ...

  5. (2015大作业)茹何优雅的手写正则表达式引擎(regular expression engine

    貌似刚开学的时候装了个逼,和老师立了个flag说我要写个正则表达式引擎,然后学期末估计老师早就忘了这茬了,在历时3个月的懒癌发作下,终于在这学期末deadline的时候花了一个下午加晚上在没有网的房间 ...

  6. Snmp的学习总结——Snmp的基本概念

    摘自:http://www.cnblogs.com/xdp-gacl/p/3978825.html 一.SNMP简单概述 1.1.什么是Snmp SNMP是英文"Simple Network ...

  7. 作诗(bzoj 2821)

    Description 神犇SJY虐完HEOI之后给傻×LYD出了一题:SHY是T国的公主,平时的一大爱好是作诗.由于时间紧迫,SHY作完诗 之后还要虐OI,于是SHY找来一篇长度为N的文章,阅读M次 ...

  8. 蒲公英(bzoj 2724)

    Description Input 修正一下 l = (l_0 + x - 1) mod n + 1, r = (r_0 + x - 1) mod n + 1 Output Sample Input ...

  9. Nearest Common Ancestors(poj 1330)

    题意:给定一棵树,询问两个节点的最近公共祖先. 输入:第一行T,表示测试组数. 每组测试数据包含一个n,表示节点数目,下面n-1行是连接的边,最后一行是询问 输出:共T行,代表每组的测试结果 /* 倍 ...

  10. 转载 cc、gcc、g++、CC的区别概括

    gcc是C编译器:g++是C++编译器:linux下cc一般是一个符号连接,指向gcc:gcc和g++都是GUN(组织)的编译器.而CC则一般是makefile里面的一个名字,即宏定义,嘿,因为Lin ...