C语⾔printf函数输出格式%x和%p的差别
%p输出sizeof(void*)⼤⼩的数据
%x输出sizeof(unsigned int)⼤⼩的数据
请不要使⽤ %x 作为指针
我经常看到这样的代码:
printf(“foo 的地址:%x\n”,&foo);
这⾥的⽬的是以⼗六进制格式打印地址。好计划; 糟糕的执⾏。
⾸先,这是⼀个可能的输出:
foo 的地址:123456
这是⼗进制 (%u)、⼗六进制 (%x) 还是⼋进制 (%o)?我⽆法从输出中分辨出来;据我所知,写那⾏的⼈是⼀个真正的新⼿,使⽤了 %i 或%d。(对于⾮常⾼的地址,情况会更糟,因为它们将被格式化为负数。)
所以作者将这⼀⾏改为:
printf(“foo 的地址:0x%x\n”,&foo);
现在输出有 0x 前缀,表明它是⼗六进制的,但这⾥仍然有⼀个错误。错误在于 %x 不是正确的指针格式化程序。
%x 预期的类型是unsigned int. 在过去⼗年左右的时间⾥,这不是问题,因为在所有 PC 上,包括 Mac,指针的⼤⼩都等于 int 的⼤⼩。
但在过去⼏年和未来⼏年,还有正在进⾏LP64,其中转接long intS和pointers是64位。普通ints 不会;它们仍然是 32 位。这意味着如果您将 %x ⽤作指针,您将获得时髦的结果,并且可能会崩溃(使⽤ %s、%n 和 %@)。
如您所见,“%p”和“%X”的输出结果⾮常相似。这种相似性会导致代码不准确,进⽽导致将程序移植到 64 位平台时发⽣错误。⼤多数情况下,使⽤“%X”⽽不是“%p”来输出指针的值,如果对象位于地址空间的四个较低有效字节之外,这将导致打印错误的值。让我们考虑⼀下这个程序对应的64 位版本:
size_t Gb =1024*1024*1024;
char*a =(char*)malloc(2* Gb *sizeof(char));
sizeof 指针char*b =(char*)malloc(2* Gb *sizeof(char));
printf("use %%X: a=%X\n", a);
printf("use %%X: b=%X\n", b);
printf("use %%p: a=%p\n", a);
printf("use %%p: b=%p\n", b);
use %X: a=80000040
use %X: b=40010040
use %p: a=0000000080000040
use %p: b=0000000140010040
使⽤“%X”时,指针值“b”打印不正确。
您可以使⽤ %lx ( unsigned long int),但它仍然是错误的类型。有⼀个专门⽤于指针的格式化程序:%p。它甚⾄为您提供“0x”前缀。printf(“foo 的地址:%p\n”,&foo);
foo 的地址:0x1e240
因此,为了使您的代码更稳定且更具前瞻性,请使⽤ %p ⽽不是 %x 或 %lx 来格式化您的指针。

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。