Printf可变參数使用
參考文档: http://bbs.csdn.net/topics/70288067
Owed by: 春夜喜雨 http://blog.csdn.net/chunyexiyu 转载请标明来源
本文的二个重点:
1. 可变參数实际上通过首个參数的地址来获取其他參数的地址。由于是顺序存储传过来的
1. 可变參数为了处理方便。全部的浮点型都是依照double型压栈。
因此,像printf採用的可变參数,它直接使用%f显示double型。不须要管float型。
关注printf的精度,须要先关注printf的实现,关于printf的实现,我们就要关注一下可变參数的实现:
变參的一个例了:
|
void { va_list argptr; va_start(argptr, if (type == 0 ) { int n = va_arg( printf( "%d\n", } else if (type == 1) { double d = printf( "%f\n", } else {} } 測试: float testIntAndFloat(0, testIntAndFloat(1, 输出: 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 printf("%d,%d \n",i, 结果为: 123456, 0 printf("%d,%d,%d,%d \n",i, 结果为: 123456,0,123456,0 printf("%lld,%lld \n",i, 结果为: 123456, 123456 |
从上面的内容能够看出来:
%d时,每次读取的位数是4位,而int64有8位。所以假设使用%d的话,分两次读取完。
(因为是little endian。先读低位,再读高位,先读到123456。再读到0)
再看输出%f的样例
|
float double fd = f; double printf("%f\%f\n%f\n",f,fd, 结果为: 1234567939550609400.0000 1234567939550609400.0000 1234567890123456800.0000 |
能够看出:
float实际是被转成double型存储显示的
(float的精底一般是6-7位。double是15-16位,printf的时候,显示的位数是按double算的。)
再看这个样例,执行时观察内存信息:
(观察内存方式: VC->调试->窗体->内存)
|
void { va_list argptr; va_start(argptr, for (int i=0; { int* pNumber = (int*)argptr; float* pFloat = (float*)argptr; va_arg(argptr, printf(("\naddrss:%d int value: %d float value: %f"), } } void { printf("\nf1=%f", printf(("\naddrss of arg(int): %x, value: %d"), &n1, printf(("\naddrss of arg(float): %x, value: %d|%d, float:%f"), &f1, printf(("\naddrss of arg(double): %x, value: %d|%d, float:%f"), &d1, printf(("\naddrss of arg(char): %x, value: %d"), &c1, printf(("\naddrss of arg(bool): %x, value: %d"), &b1, printf(("\naddrss of arg(short): %x, value: %d"), &s1, printf(("\naddrss of arg(__int64): %x, value: %d"), &n2, printf(("\naddrss of arg(int): %x, value: %d"), &n3, testArgs(n1, } |
当按函数定參參数形式传递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
Printf可变參数使用的更多相关文章
- C语言利用va_list、va_start、va_end、va_arg宏定义可变參数的函数
在定义可变參数的函数之前,先来理解一下函数參数的传递原理: 1.函数參数是以栈这样的数据结构来存取的,在函数參数列表中,从右至左依次入栈. 2.參数的内存存放格式:參数的内存地址存放在内存的堆栈段中, ...
- java课堂练习之可变參数与卫条件
/* 有人邀请A,B,C,D,E,F 6个人參加一项会议,这6个人有些奇怪.由于他们有非常多要求,已知: 1)A,B两人至少有1人參加会议: 2)A,E,F 3人中有2人參加会议. 3)B和C ...
- OC可变參数的函数实现va_start、va_end、va_list的使用
一.简单介绍 我们常常在编程的时候看见类似这种代码,如图1.1 图1.1 或者是这种可变參数,如图1.2 图1.2 二.基本知识介绍 在学习怎样写这样的格式的函数前,先简介几个经常使用的宏: 下面摘自 ...
- iOS 处理方法中的可变參数
## iOS 处理方法中的可变參数 近期写了一个自己定义的对话框的demo,想模仿系统的UIAlertView的实现方式.对处理可变參数的时候,遇到了小问题,于是谷歌了一下.写下了处理问题的方法.记录 ...
- Effective JavaScript Item 21 使用apply方法调用函数以传入可变參数列表
本系列作为Effective JavaScript的读书笔记. 以下是一个拥有可变參数列表的方法的典型样例: average(1, 2, 3); // 2 average(1); // 1 avera ...
- java 可变參数
我们在某些特定的需求环境下,可能要对某一个方法中的參数进行一些操作,并且这些方法中的參数是不规定的,那么问题来了,我们该怎么办呢? java事实上就为我们考虑了这样的情况,那就是使用可变參数 可变參数 ...
- java 可变參数列表
Java SE5加入了可变參数列表特性 參数能够这样定义.(Object-args).可变參数用"..."来定义,args是可变參数的数组.举个样例: package sample ...
- java之 ------ 可变參数和卫条件
可变參数:适用于參数个数不确定.类型确定的情况,java把可变參数当做数组处理. 可变參数必须位于最后一项.当可变參数个数多于一个时,必将有一个不是最后一项,所以仅仅支持有一个可变參数. 可变參数的书 ...
- 增强for循环、Map接口遍历、可变參数方法
增强for循环 1.for循环能做得事情.增强for循环大部分都能做(假设要想获得下标的时候就必须使用简单for循环了) 2.增强for有时候可以方便的处理集合遍历的问题,可是集合的标准遍历是使用迭代 ...
随机推荐
- CSAPP学习笔记—虚拟内存
CSAPP学习笔记—虚拟内存 符号说明 虚拟内存地址寻址 图9-12展示了MMU如何利用页表来实现这种映射.CPU中的一个控制寄存器,页表基址寄存器(Page Table Base Register, ...
- document.execCommand
document.execCommand 在firefox浏览器执行不好,但是在其他浏览器有时候使用会非常方便. 比如在input标签中使用: onkeyup="if(isNaN(value ...
- 如何部署 sources and javadoc jars
mvn org.apache.maven.plugins:maven-deploy-plugin:2.8.2:deploy-file -Durl=file:///home/me/m2-repo \ - ...
- 通过 Terracotta实现基于Tomcat的Web应用集群
[转]通过 Terracotta实现基于Tomcat的Web应用集群 博客分类: 企业应用面临的问题 Java&Socket 开源组件的应用 tomcatweb session集群服务器负载均 ...
- (2015大作业)茹何优雅的手写正则表达式引擎(regular expression engine
貌似刚开学的时候装了个逼,和老师立了个flag说我要写个正则表达式引擎,然后学期末估计老师早就忘了这茬了,在历时3个月的懒癌发作下,终于在这学期末deadline的时候花了一个下午加晚上在没有网的房间 ...
- Snmp的学习总结——Snmp的基本概念
摘自:http://www.cnblogs.com/xdp-gacl/p/3978825.html 一.SNMP简单概述 1.1.什么是Snmp SNMP是英文"Simple Network ...
- 作诗(bzoj 2821)
Description 神犇SJY虐完HEOI之后给傻×LYD出了一题:SHY是T国的公主,平时的一大爱好是作诗.由于时间紧迫,SHY作完诗 之后还要虐OI,于是SHY找来一篇长度为N的文章,阅读M次 ...
- 蒲公英(bzoj 2724)
Description Input 修正一下 l = (l_0 + x - 1) mod n + 1, r = (r_0 + x - 1) mod n + 1 Output Sample Input ...
- Nearest Common Ancestors(poj 1330)
题意:给定一棵树,询问两个节点的最近公共祖先. 输入:第一行T,表示测试组数. 每组测试数据包含一个n,表示节点数目,下面n-1行是连接的边,最后一行是询问 输出:共T行,代表每组的测试结果 /* 倍 ...
- 转载 cc、gcc、g++、CC的区别概括
gcc是C编译器:g++是C++编译器:linux下cc一般是一个符号连接,指向gcc:gcc和g++都是GUN(组织)的编译器.而CC则一般是makefile里面的一个名字,即宏定义,嘿,因为Lin ...