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));
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 来格式化您的指针。

  • 5
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值