当指向特定类型(例如int, char, float, ..)的指针增加时,其值将按该数据类型的大小增加。如果指向大小为x的数据的空指针被增加,它如何提前x字节指向?编译器如何知道将x添加到指针的值?
当前回答
C标准不允许空指针运算。然而,考虑到void的大小为1,GNU C是允许的。
C11标准§6.2.5
第19段
void类型由一组空值组成;这是一个不完整的 无法完成的对象类型。
以下程序在GCC编译器中工作良好。
#include<stdio.h>
int main()
{
int arr[2] = {1, 2};
void *ptr = &arr;
ptr = ptr + sizeof(int);
printf("%d\n", *(int *)ptr);
return 0;
}
可能是其他编译器产生错误。
其他回答
不允许对void*指针进行指针算术运算。
C标准不允许空指针运算。然而,考虑到void的大小为1,GNU C是允许的。
C11标准§6.2.5
第19段
void类型由一组空值组成;这是一个不完整的 无法完成的对象类型。
以下程序在GCC编译器中工作良好。
#include<stdio.h>
int main()
{
int arr[2] = {1, 2};
void *ptr = &arr;
ptr = ptr + sizeof(int);
printf("%d\n", *(int *)ptr);
return 0;
}
可能是其他编译器产生错误。
将其转换为char型指针,使指针向前增加x字节。
编译器通过类型转换知道。给定一个void *x:
X +1向X增加一个字节,指针指向字节X +1 (int*)x+1添加sizeof(int)字节,指针指向字节x+ sizeof(int) (float*)x+1地址sizeof(float)字节, 等。
尽管第一项是不可移植的,并且反对C/ c++的Galateo,但它仍然是C语言正确的,这意味着它将在大多数编译器上编译成可能需要适当标志的东西(如-Wpointer-arith)
Void指针可以指向任何内存块。因此,当我们尝试对空指针进行指针算术时,编译器不知道要增加/减少多少字节。因此,必须首先将void指针类型转换为已知类型,然后才能将它们涉及到任何指针算术。
void *p = malloc(sizeof(char)*10);
p++; //compiler does how many where to pint the pointer after this increment operation
char * c = (char *)p;
c++; // compiler will increment the c by 1, since size of char is 1 byte.
推荐文章
- C多行宏:do/while(0) vs作用域块
- time_t最终的类型定义是什么?
- 我需要显式处理负数或零时,总和平方数字?
- 函数名周围的括号是什么意思?
- 用C语言创建自己的头文件
- 格式化IO函数(*printf / *scanf)中的转换说明符%i和%d之间的区别是什么?
- main()中的Return语句vs exit()
- 如果不是内存地址,C指针到底是什么?
- 我如何在Visual Studio中预处理后看到C/ c++源文件?
- 保护可执行文件不受逆向工程的影响?
- 从C语言的函数返回一个struct
- C99 'restrict'关键字的实际用法?
- 检查子字符串存在于C中的字符串中
- 从标准输入中捕获字符,而不需要等待按enter键
- 查找当前可执行文件的路径,不包含/proc/self/exe