1、深入理解 C语言指针的奥秘 原 作 者:不祥 原 出 处: vcer 发 布 者:李广胜 发布类型:转载 发布日期: 2007-12-18 今日 /总浏览: 10/1177 指针是一个特殊的变量,它里面存储的数值被解释成为内存里的一个地址。 要搞清一个指针需要搞清指针的四方面的内容:指针的类型,指针所指向的 类型,指针的值或者叫指针所指向的内存区,还有指针本身所占据的内存区。让我们分别说明。 先声明几个指针放着做例子: 例一: (1)int*ptr; (2)char*ptr; (3)int*ptr; (4)int(*ptr)3; (5)int*(*ptr)4; 指针的类型 从语法的角度看,你只
2、要把指针声明语句里的指针名字去掉,剩下的部分就是这个指针的类型。这是指针本身所具有的类型。让我们看看例一中各个指针的类型: (1)int*ptr;/指针的类型是 int* (2)char*ptr;/指针的类型是 char* (3)int*ptr;/指针的类型是 int* (4)int(*ptr)3;/指针的类型是 int(*)3 (5)int*(*ptr)4;/指针的类型是 int*(*)4 怎么样?找出指针的类型的方法是不是很简单? 指针所指向的类型 当你通过指针来访问指针所指向的内存区时,指针所指向的类型决定了编译器将把那片内存区里的内容当做什么来看待。 从语法上看,你只须把指针声明语句中
3、的指针名字和名字左边的指针声明符 *去掉,剩下的就是指针所指向的类型。例如: (1)int*ptr;/指针所指向的类型是 int (2)char*ptr;/指针所指向的的类型是 char (3)int*ptr;/指针所指向的的类型是 int* (4)int(*ptr)3;/指针所指向的的类型是 int()3 (5)int*(*ptr)4;/指针所指向的的类型是 int*()4 在指针的算术运算中,指针所指向的类型有很大的作用。 指针的类型 (即指针本身的类型 )和指针所指向的类型是两个概念。当你对 C越来越熟悉时,你会发现,把与指针搅和在一起的 “类型 “这个概念分成 “指针的类型 “和 “指
4、针所指向的类型 “两个概念,是精通指针的关键点之一。我看了不少书,发现有些写得差的书中,就把指针的这两个概念搅在一起了,所以看起书来前后矛盾,越看越糊涂。 指针的值 ,或者叫指针所指向的内存区或地址。指针的值是指针本身存储的数值,这个值将被编译器当作一个地址,而不是一个一般的数值。在 32 位程序里,所有类型的指针的值都是一个 32位整数,因为 32位程序里内存地址全都是 32位长。 指针所指向的内存区就是从指针的值所代表的那个内存地址开始,长度为 si zeof(指针所指向的类型 )的一片内存区。以后,我们说一个指针的值是 XX,就相当于说该指针指向了以 XX 为首地址的一片内存区域;我们说
5、一个指针指向了某块内存区域,就相当于说该指针的值是这块内存区域的首地址。 指针所指向的内存区和指 针所指向的类型是两个完全不同的概念。在例一中,指针所指向的类型已经有了,但由于指针还未初始化,所以它所指向的内存区是不存在的,或者说是无意义的。以后,每遇到一个指针,都应该问问:这个指针的类型是什么?指针指的类型是什么?该指针指向了哪里? 指针本身所占据的内存区 指针本身占了多大的内存?你只要用函数 sizeof(指针的类型 )测一下就知道了。在 32位平台里,指针本身占据了 4个字节的长度。 指针本身占据的内存这个概念在判断一个指针表达式是否是左值时很有用。 指针的算术运算 指针可以加上或减去一
6、个整 数。指针的这种运算的意义和通常的数值的加减运算的意义是不一样的。例如: 例二: 1、 char a20; 2、 int*ptr=a; . . 3、 ptr+; 在上例中,指针 ptr 的类型是 int*,它指向的类型是 int,它被初始化为指向整形变量 a。接下来的第 3 句中,指针 ptr被加了 1,编译器是这样处理的:它把指针 ptr的值加上了 sizeof(int),在 32位程序中,是被加上了 4。由于地址是用字节做单位的,故 ptr 所指向的地址由原来的变量 a 的地址向高地址方向增加了 4 个字节。 由于 char 类型的长度是一个字节,所以,原来 ptr是指向数组 a 的第
7、 0号单元开始的四个字节,此时指向了数组 a 中从第 4号单元开始的四个字节。 我们可以用一个指针和一个循环来遍历一个数组,看例子: 例三: Int array20; Int *ptr=array; . /此处略去为整型数组赋值的代码。 . for(i=0;ia; ptr-b; ptr-c; 又请问怎样通过指针 pstr来访问 ss的三个成员变量? 答案: *pstr; /访问了 ss的成员 a。 *(pstr+1);/访问了 ss的成员 b。 *(pstr+2)/访问了 ss的成员 c。 虽然我在我的 MSVC+6.0上调式过上述代码,但是要知道,这样使用 pstr来访问结构成员是不正规的,为了说明为什么不正规,让我们看看怎样通过指针来访问数组的各个单元: 例十二: intarray3=35,56,37; int*pa=array; 通过指针 pa 访问数组 array 的三个单元的方法是: *pa;/访问了第 0号单元