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有时候可以方便的处理集合遍历的问题,可是集合的标准遍历是使用迭代 ...
随机推荐
- 九度oj 题目1153:括号匹配问题
题目描述: 在某个字符串(长度不超过100)中有左括号.右括号和大小写字母:规定(与常见的算数式子一样)任何一个左括号都从内到外与在它右边且距离最近的右括号匹配.写一个程序,找到无法匹配的左括号和右括 ...
- 01-offsetWidth和offsetHeight
<!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8&quo ...
- 对拍程序(Win)
代码如下: @echo off :again rand.exe echo "rand finish" asd.exe echo "1.exe finish" 未 ...
- [BZOJ1575] [Usaco2009 Jan]气象牛Baric(DP)
传送门 DP f[i][j]表示前i个中选j个的最优解 预处理g[i][j]表示选i~j对答案的贡献 那么就可以n^3乱搞了! 注意边界 #include <cstdio> #includ ...
- BZOJ1833 [ZJOI2010]count 数字计数 【数学 Or 数位dp】
题目 给定两个正整数a和b,求在[a,b]中的所有整数中,每个数码(digit)各出现了多少次. 输入格式 输入文件中仅包含一行两个整数a.b,含义如上所述. 输出格式 输出文件中包含一行10个整数, ...
- 刷题总结——郁闷的出纳员(bzoj1503)
题目: 题目背景 NOI2004 DAY1 T1 题目描述 OIER 公司是一家大型专业化软件公司,有着数以万计的员工.作为一名出纳员,我的任务之一便是统计每位员工的工资.这本来是一份不错的工作,但是 ...
- 【leetcode dp】Dungeon Game
https://leetcode.com/problems/dungeon-game/description/ [题意] 给定m*n的地牢,王子初始位置在左上角,公主在右下角不动,王子要去救公主,每步 ...
- Ionic1与Ionic2
1.Ionic2新特性 ①组织结构与框架: 在Ionic2中,每个组件.页面都只专注于做一件事,它单独有自己的一个目录,有自己的类(Class).模板文件(Template)和自己的样式文件(在这里我 ...
- hdu5396 Expression
Expression Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)Total ...
- 大数(bzoj 4542)
/* 想了半天莫队,不知道咋转移,需要动下脑子. 有一个很显然的结论是如果(l,r)是P的倍数,那么s[l...n]%P=s[r+1...n]%P. 根据这个东西,我们预处理出所有的后缀%P的余数,接 ...