1.指针的类型
从语法上来看,你只要把指针声明语句里的指针名字去掉,剩下的部分就是这个指针的类型,这是指针本身具有的类型。
实例
(1)int *ptr; //指针的类型是 int*
(2)char *ptr; //指针的类型是 char*
(3)int **ptr; //指针的类型是 int**
(4)int (*ptr)[3]; //指针的类型是 int (*)[3]
(5)int *(*ptr)[4]; //指针的类型是 int *(*)[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.指针的值
指针的值也叫做指针所指向的内存区域或地址。指针的值是指针本身存储的数值,这个值将被编译器当作一个地址,而不是一个一般的数值。在32位程序里,所有类型的指针的值都是一个32位整数,因为32位程序里内存地址全都是32位长。指针所指向的内存区域就是从指针值所代表的那个内存地址开始,长度为sizeof的一片内存区域。
以后我们说一个指针的值是XX,就相当于说该指针指向了以XX位首地址的一片内存区域;我们说一个指针指向了某块内存区域,就相当于说该指针的值是这块内存区域的首地址。
4.指针的算术运算
第一种是:指针 + (-) 整数
标准定义这种形式只能用于指向数组中的某个元素的指针,这类表达式的结果类型也是指针。
数组的元素存储于连续的内存位置中,后面元素的地址大于前面元素的地址,因此,对一个指针+ 1 使他指向下一个元素,+5 使他向右移动5 个元素的位置,以此类推,把一个指针- 3 使他向左移动3个元素的位置。
第二种是:指针 - 指针
只有当两个指针都指向同一个数组中的元素时,才允许从一个指针减去另一个指针。两个指针相减的结果的类型是 ptrdiff_t,他是一种有符号的整数类型。
减法运算的值是两个指针在内存中的距离,因此减法运算的结果将除以数组元素类型的长度。
5.指针的运算关系
对指针执行关系运算也是有限制的,用以下关系操作符对两个指针值进行比较是有可能的,不过前提是他们都指向同一个数组的元素。
<= , < , > , >=
当然,也可以在两个任意的指针间执行相等或不相等测试,因为这类比较结果和编译器选择在何处存储数据并无关系——指针要么指向同行一个地址,要么指向不同地址。
6.NULL总是被定义为0
NULL不是定义为0,就是被定义为“(void *)0”,这两种值几乎是相同的。当程序中需要一个指针时(尽管编译程序并不是总能只是什么时候需要一个指针),一个纯粹的0或者一个void指针都能自动被转换成所需要的任何类型的指针。
7.函数指针与指针函数
(1)函数指针:即指向这个函数的指针,定义为“数据类型*(*fun)(参数列表);”,()的优先级比*高,所以*fun加括号,如“void(*fun)(int* , int*);”
(2)指针函数:即返回值是指针的函数,定义为“数据类型*fun(参数列表);”,如“char* fun(int* , int*);”,即返回值为char*型。
8.函数地址
在C语言中,变量有它的地址,同理函数也有地址的,那么把函数的地址赋给函数指针,在通过函数指针调用这个函数就可以
第一步:定义函数指针,如“int (*pfun)(int*, int*);”。
第二步:定义函数,如“int fun(int* , int*);”。
第三步:把函数的地址赋给函数指针,即“pfun = fun;”。
第四部:通过函数指针去调用这个函数“(*pfun)(p, q);”,pfun是函数的地址,那么*pfun当然就是函数本身了。
9.函数参数传递问题
在C语言中,有两种参数传递的方式,一种是值传递,另一种是指针传递。
值传递很好理解, 即把实参的值传递给形参。
指针传递传递的是地址在C语言中,形参值的改变并不能改变实参的值,但形参所指向内容值的改变却能改变实参,这一点非常的重要,是指针传递的精华所在。
10.指针函数
当函数的返回值为指针类型时,应该尽量不要返回局部变量的指针,因为局部变量是定义在函数内部,当这个函数调用结束了,局部变量的栈内存也被释放了,因此不能够正确的得到返回值。实际上,内存已经被释放了,但这个指针的地址已经返回过去了,但是这个地址已经是无效的了,此时,对这个指针的使用是很危险的。