指针是C语言的根本,C/C++功能强大,主要体现在指针的灵活运用。灵活地应用指针,可有效地表示复杂的数据结构、可直接处理内存地址,动态分配内存、更简单有效的处理字符串和数组。本文着重介绍对指针概念的深入理解及多种用法。供初学者或相关教学的人员参考。
一、指针概念的理解
指针是一个特殊的变量,它里面存储的数值被解释成为内存里的一个地址。理解指针首先要明确指针四个方面的内容:指针的类型、指针所指向的类型、指针的值(指针所指向的内存区)和指针本身所占据的内存区。
1、指针的类型。
在该指针的定义语句中把指针变量名(pointer)去掉,就是这个指针的类型。这是指针本身所具有的类型。例如:
2、指针所指向的数据类型。
指针的类型(即指针本身的类型)和指针所指向的类型是两个概念。在通过指针来访问指针所指向的内存区时,指针所指向的类型决定了编译器将把那片内存区里的内容当做什么来看待。即指针从该内存地址开始一次取出多少字节作为一个整体进行处理。从语法上看,把指针声明语句中的指针变量名和变量名左边的指针声明符(*)去掉,就是指针所指向的类型。例如:
在指针的算术运算中,指针所指向的类型有很大的作用。例如pointer+n操作中,指针pointer增加了n*sizeof(指针所指向的类型)。
3、指针的值
指针的值是指针本身存储的数值,这个值将被编译器当作一个地址即指针所指向内存区的地址,而不是一个一般的数值。在32位程序里,所有类型的指针的值都是一个32位整数。
指针所指向的内存区就是从指针的值所代表的那个内存地址开始,长度为sizeof(指针所指向的类型)的一片内存区。通常,一个指针指向了某块内存区域,即该指针的值是这块内存区域的首地址。一个指针的值是XX,即该指针指向了以XX为首地址的一片内存区域。
指针所指向的内存区和指针所指向的类型是两个完全不同的概念。定义指针所指向的类型后,但由于指针还未初始化,所以它所指向的内存区是不存在的,也是无意义的。
4、指针本身所占据的内存空间
指针本身所占据的内存区可用函数sizeof(指针的类型)测试。在32位平台里,指针本身占据了4个字节的长度。指针本身占据的内存这个概念在判断一个指针表达式是否为左值时很有用。
5、将C程序返汇编后查看指针
C语言执行前会编译成机器语言的目标程序,而机器语言难以理解。可将C语言的源程序反汇编为汇编程序,查看指针的使用。例如:
C源程序:
main()
{chars='a';
char*p=&s;
putchar(*p);
}
反汇编后:
pushbp
movbp,sp
subsp,2
pushsi
movbyteptr[bp-1],97
leasi,wordptr[bp-1];si即指针p
moval,byteptr[si]
cbw
pushax
callnearptr_putchar
popcx
@1:
popsi
movsp,bp
popbp
ret
可以看出SI即是指针p,汇编语言通过lea指令完成指针的赋值。用寄存器间接寻址实现指针运算*p。在简单的指针运算中,用寄存器表示指针,复杂的指针则放入内存。
二、指针的使用
1、空指针
空指针在指针和链表中广泛使用,表示不指向任何对象的一种指针或分配失败。其值定义为NULL,它是在头文件“stdio.h”中定义的一个宏,其值与任何有效指针的值都不同,NULL是一个纯粹的零(段0,偏移量0),它可能被强制转换为void*或char*类型。即NULL可能是0或(void*)0等。
值得注意的是,绝对不能间接引用一个空指针,否则,程序可能会得到一个毫无意义的结果,或得到一个全部是零的值,甚至突然停止运行。
空指针和未初始化的指针是连两个截然不同的概念。空指针已经被明确不指向任何对象或函数,未初始化的指针可以指向任何位置。不同类型的空指针的内在的值可能是不同的。虽然编程者不需要知道其值,但是必须告知编译器需要的是何种类型的空指针。所以在必要的时候应加以区别。根据语法规定,一个指向常数0的指针在编译时会转化为空指针。因此在初始化、赋值或比较表达式中,一边是指针类型的变量或表达式,另一边是常数0,编译器则会将0转化成一个空指针,并生成一个确切类型的空指针的值。例如下列语句是完全合法的:char*p=0;if(p!=0)……
类似C/C++字符串以’