注册 登录
编程论坛 C语言论坛

C语言中,指针相减与右移

八画小子 发布于 2020-07-08 23:20, 2227 次点击
刚刚踩了一个坑,等效代码如下:
程序代码:
#include <stdio.h>
#include <stdlib.h>

int main(void)
{
    void * start = (void *)0xc0000000;
    void * end = (void *)0x00000000;

    printf("%p - %p = %lx\n", end, start, end - start);
    printf("(%p - %p) >> 12 = %#lx\n", end, start, (end - start) >> 12);
    printf("(unsigned long)(%p - %p) >> 12 = %#lx\n", end, start, (unsigned long)(end - start) >> 12);

    return EXIT_SUCCESS;
}


结果(end - start) >> 12这一部分的计算有问题,导致了一个错误。

C语言中的右移动操作符>>视操作数是否为无符号有两种操作:如果操作数是无符号,则按逻辑右移执行;如果操作数是有符号,则按算术右移执行。
这一点是没有问题的,不过指针相减的结果却是有符号整数,故(end - start) >> 12得不到预期的结果,应该做一次类型转换。
3 回复
#2
rjsp2020-07-09 00:16
指针相减的类型是 ptrdiff_t,是有符号类型,在printf中用 %td。
可以转为 uintptr_t 等无符号类型,在printf中用 <inttypes.h> 中定义的 PRIuPTR。
#3
rjsp2020-07-09 00:19
你想16进制输出uintptr_t,那么应该用 PRIxPTR、PRIXPTR
#4
八画小子2020-07-09 01:08
以下是引用rjsp在2020-7-9 00:16:30的发言:

指针相减的类型是 ptrdiff_t,是有符号类型,在printf中用 %td。
可以转为 uintptr_t 等无符号类型,在printf中用 <inttypes.h> 中定义的 PRIuPTR。



受教了,标准库里确实有很多比较方便的东西,研读标准库是非常有必要的。👍👍👍
1