printf使用隐患

平时习惯使用iostream中的cout,很久不用printf, sprintf, fprintf等格式化输出的方法了,不过,最近的一次使用,发现自己一个错误的用法,造成安全隐患。

longbyte a;

float b;

...

printf("a: %d, b: %f\n", a, b);


a是一个长字节的自定义类型数据,b是一个float型数据。日志里,输出的结果都不正常,经排查,不是a, b数据的原因,而是printf的使用方法不当。


printf格式化输出,仅仅是将数据按照给定格式输出,并不进行数据类型的隐式转换,是利用 二进制的方式格式化为字符串。由于a是一个长字节数据,而其格式化为整型4个字节输出,不仅如此,更令人郁闷的是,它还影响到b的输出,使b的输出完全不对。


所以,printf要求输出格式类型与数据类型完全一致。至于,溢出的后果,查阅了,没有统一说明,有的说,标准对结果未定义,不同的编译器可能有不同的结果。

这么不安全的用法,以后尽量避免它吧,否则,小心对待。


-------------------------


记录个数据类型隐式转换,截断的问题。


char m = 0xeeff; 

int n = 0xeeff;

char m = n;

m的结果是0xff,这些都是语法逻辑上的截断使用方法。大部分的机器上运行的结果该相同。


但若

int a = 0xeeff;

char *b = &a;

*b的结果,在不同的机器上,其运行结果就不一定相同了。

原因是机器的数据存储方式上分为大端法(big endian)与小端法(little endian),这个用法的在网络传输编程或底层编程要求的比较多。

big endian将数据高位放在低地址处,网络传输一般都用big endian方式,所以,要与本机字节进行转换。

而little endian将数据高位却放在高地址处。


使用取地址的方法对数据进行截断,就要考虑数据在机器上的存储方式了。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值