每遇到一个指针,都应该问问:这个指针的类型是什么?指针指向的类型是什么?该指针指向了哪里?
int *ptr; ?指针所指向的类型已经有了,但由于指针还未初始化,所以它所指向的内存区是不存在的,或者说是无意义的。
//指针的特点: void main() {
?? ?int* ptr; ?? ?printf("sizeof(int*)=%d\n", sizeof(int*));//sizeof(int*)=8 ?? ?printf("sizeof(int)=%d\n", sizeof(int));//sizeof(int)=4 ?? ?int a = 23; ?? ?ptr = &a; ?? ?printf("*ptr=%d\n", *ptr);//*ptr=23 ?? ?printf("ptr=%d\n", ptr);//ptr=1821375092,这是随机的
?? ?printf("++*ptr=%d\n", ++*ptr);//*ptr=23+1 ?? ?printf("--ptr=%d\n", --ptr);//ptr=1821375092-4 }
指针是一个特殊的变量,它里面存储的数值被解释成为内存里的一个地址。
要搞清一个指针需要搞清指针的四方面的内容:指针的类型,指针所指向的类型,指针的值或者叫指针所指向的内存区,还有指针本身所占据的内存区。
(只要把指针声明语句里的指针名字去掉,剩下的部分就是这个指针的类型。这是指针本身所具有的类型。)
int *ptr; //指针的类型是int * ?
char *ptr; //指针的类型是char * ?
int **ptr; //指针的类型是 int **?
int (*ptr)[3]; //指针的类型是 int(*)[3]?
int *(*ptr)[4]; //指针的类型是 int *(*)[4] ?
指针所指向的类型
通过指针来访问指针所指向的内存区时,指针所指向的类型决定了编译器将把那片内存区里的内容当做什么来看待。
从语法上看,只须把指针声明语句中的指针名字和名字左边的指针声明符*去掉,剩下的就是指针所指向的类型。
int *ptr; //指针所指向的类型是int
char *ptr; //指针所指向的的类型是char
int **ptr; //指针所指向的的类型是 int *
int (*ptr)[3]; //指针所指向的的类型是 int()[3]
int *(*ptr)[4]; //指针所指向的的类型是 int *()[4]
在指针的算术运算中,指针所指向的类型有很大的作用。
指针的类型(即指针本身的类型)和指针所指向的类型是两个概念。
指针的值是指针本身存储的数值,这个值将被编译器当作一个地址,而不是一个一般的数值。在32位程序里,所有类型的指针的值都是一个32位整数,因为32位程序里内存地址全都是32位长。
指针所指向的内存区就是从指针的值所代表的那个内存地址开始,长度为sizeof(指针所指向的类型)的一片内存区。以后,我们说一个指针的值是XX,就相当于说该指针指向了以XX为首地址的一片内存区域;我们说一个指针指向了某块内存区域,就相当于说该指针的值是这块内存区域的首地址。
指针所指向的内存区和指针所指向的类型是两个完全不同的概念。
在例一中,int *ptr; ?指针所指向的类型已经有了,但由于指针还未初始化,所以它所指向的内存区是不存在的,或者说是无意义的。
|