C语言中的指针-2
指针数组与数组指针
指针数组
由指针变量组成的数组,它的身份是数组,成员是指针变量。
类型 * arr[n];
int * arr[10];
数组指针
它的身体是指针,专门用来存储数组的地址。
类型( * 变量名)[n];
int ( * arr) [5];
注意:数组指针是以数组字节宽度为移动单位。
函数指针
函数会翻译二进制指令存储到代码段,而函数名就是这段指令的首地址。
可以定义特殊的指针变量存储函数的首地址,这样就可以把函数当前数据进行传递。
我们把存储函数地址的指针变量叫做函数指针。
如何定义函数指针:
1.复制函数的声明语句
2.给函数名加上小括号并在函数名前加*
3.给函数名重新取函数指针变量名
由于函数指针的类型比较长,一般使用typedef 进行类型重定义。
void func(int,int);
typedef void (*FP)(int,int);
二级指针
一级指针中存储的是普通变量的地址,二级指针里面存储的是一级指针变量的地址。
当跨函数共享指针变量时,需要使用二级指针。
定义二级指针:
类型 * * 变量名 ;
二级指针的赋值和解引用
int num;
int* p = #
int** pp =&p;
*pp <=>p;
**pp <=>*p;
指针与数组名的关系
数组名就是数组空间的首地址,是一个常量地址,可以把它看做常指针。
数组名可以使用解引用的语法访问成员,指针也可以使用[]解引用,所以:
? arr[n] <=> * (arr+n);
与普通指针的区别
1.数组名是常量,普通指针是变量
2.普通指针变量有自己的存储空间,用来存储内存编号,它与目标内存是指向关系。
3.数组名没有自己的存储空间,它就代表数组空间的首地址,它与数组内存是映入关系。
4.对数组名取地址结果还是数组名的值
int arr[5]
arr <=> &arr;
arr 类型 int*
&arr 类型 int (*)[5]
使用指针要注意的问题
空指针
指针变量的值为NULL的指针被称为空指针,这种指针不能解引用,否则会产生段错误。
大多数系统的NULL是0地址,可以把指针变量当做逻辑值来使用,但不建议这样。
为了避免空指针导致的段错误,当使用来历不明的指针时应判断是否是空指针。
野指针
指针变量的值是随机的、不确定的,这种指针被称为野指针。使用野指针不一定立即出错,也不一定会出错,与空指针相比它的危害性较大。
无法判断一个指针变量是否是野指针,即野指针在大量的代码中不易找出来,基本上找不出来,所以为了避免野指针导致的bug,我们应避免产生野指针。
1.定义指针变量时一定要初始化。
2.指针变量所指向的内存被释放、销毁后,指针变量要及时赋值为NULL。
3.函数不返回局部变量的地址。
|