在上一篇c专题指针文章中,我们介绍了什么是指针,文章里面从普通变量进而引出指针的概念,这样对指针的理解有一定的帮助(其实最好的理解,就是要明白硬件里面的内存原理,这是理解指针最好的地方,就好比说会汇编语言的人来去理解指针这里跟不会指针的人去理解,会有很大的差异的,在学汇编的时候,会接触到好多有关计算机里面内存的大话题,这个对于搞汇编的来说,掌握了汇编,对理解指针的原理非常容易;而大部分人(当然也包括我自己),刚开始学指针,是真的非常吃力,学了一阵子,感觉是学会了,但是一段时间没有去接触指针,再次来看指针的话,感觉一脸懵逼,好像没学过一样,不知道大家有没有我这样的经历,哈哈哈;这里指出不是鼓励大家去学花太多时间在汇编上(个人观点,现在出来上班,好少会搞汇编开发,你搞stm32和一些稍微功能强大的芯片,拿汇编去写,那简直不敢想象,而且也没听过谁这样干过),其实还是当你用到的时候再去学,很快上手的,就是有好多汇编指令要记,如果你一遍学一遍用,反而会学的更快,理解的更深,而且现在对理解一些高级芯片里面的启动代码会非常有帮助的)。好了,废话太多,来进入主题!
一、空指针:
1、什么是空指针?
在C语言中,如果一个指针不指向任何数据,我们就称之为空指针,用NULL表示,例如:
代码语言:javascript复制 int *a = NULL;
NULL在C/C 中定义为:
代码语言:javascript复制 #ifdef _cplusplus // 定义这个符号就表示当前是C 环境
#define NULL 0 // 在C 中NULL就是0
#else
#define NULL (void *)0 // 在C中NULL是强制类型转换为void *的0
#endif
说明:a、在C语言中,int *p;你可以p = (int *)0;但是不可以p = 0;因为类型不相同。所以NULL的实质其实就是0,然后我们给指针赋初值为NULL,其实就是让指针指向0地址处。为什么指向0地址处?2个原因。第一层原因是0地址处作为一个特殊地址(我们认为指针指向这里就表示指针没有被初始化,就表示是野指针);第二层原因是这个地址0地址在一般的操作系统中都是不可被访问的,如果C语言程序员不按规矩(不检查是否等于NULL就去解引用)写代码直接去解引用就会触发段错误(下面讲野指针有举例子),这种已经是最好的结果了。
b、一般在判断指针是否野指针时,都写成if (NULL != p)而不是写成 if (p != NULL)原因是:如果NULL写在后面,当中间是==号的时候,有时候容易忘记写成了=,这时候其实程序已经错误,但是编译器不会报错。这个错误(对新手)很难检查出来;如果习惯了把NULL写在前面,当错误的把==写成了=时,编译器会报错,程序员会发现这个错误(这里自己昨天就在这里犯了低级错误)。
一、野指针:
1、什么是野指针?
简单来讲它就是指针指向的位置是不可知的(随机的、不正确的、没有明确限制的),想必有一定基础的好友,一定看过这样的代码:
代码语言:javascript复制 #include <stdio.h>
int main()
{
int *a;
*a=9;
return 0;
}
说明:这里是在gcc编译器下,会出现段错误(Sgmentation fault);但是在dev--c 上是不会报错的。因为指针变量在定义时如果未初始化,值也是随机的。指针变量的值其实就是别的变量(指针所指向的那个变量)的地址,所以意味着这个指针指向了一个地址是不确定的变量,这时候去解引用就是去访问这个地址不确定的变量,所以结果是不可知的。
2、野指针的危害:
a、指向不可访问(操作系统不允许访问的敏感地址,譬如内核空间)的地址,结果是触发段错误,这种算是最好的情况了。
b、指向一个可用的、而且没什么特别意义的空间(譬如我们曾经使用过但是已经不用的栈空间或堆空间),这时候程序运行不会出错,也不会对当前程序造成损害,这种情况下会掩盖你的程序错误,让你以为程序没问题,其实是有问题的。
c、指向了一个可用的空间,而且这个空间其实在程序中正在被使用(譬如说是程序的一个变量x),那么野指针的解引用就会刚好修改这个变量x的值,导致这个变量莫名其妙的被改变,程序出现离奇的错误。一般最终都会导致程序崩溃,或者数据被损害。这种危害是最大的。
------小结: 指针变量如果是局部变量,则分配在栈上,本身遵从栈的规律(反复使用,使用完不擦除,所以是脏的,本次在栈上分配到的变量的默认值是上次这个栈空间被使用时余留下来的值),就决定了栈的使用多少会影响这个默认值。所以我们要避免这种情况发生。
3、怎样来避免野指针的出现?
野指针的错误来源就是指针定义了以后没有初始化,也没有赋值(总之就是指针没有明确的指向一个可用的内存空间),然后去解引用。因此个人推荐大家一般常用的方法:
第一点:定义指针时,同时初始化为NULL
第二点:在指针解引用之前,先去判断这个指针是不是NULL
第三点:指针使用完之后,将其赋值为NULL
第四点:在指针使用之前,将其赋值绑定给一个可用地址空间
代码语言:javascript复制 #include <stdio.h>
int main()
{
int a;
int *b=NULL;
b=&a;// 正确的使用指针的方式,是解引用指针前跟一个绝对可用的地址绑定
if(NULL != b)
{
*b=8;
}
b=NULL;
return 0;
}
注明:一般来说,定义一个指针都会给初始化一个有效地址来操作。
三、总结:
现在看完文章,你应该对指针的使用一定要注意给初始化,否则就会出现问题。再次强调一下:void 指针与空指针 NULL 不同,NULL 说明指针不指向任何数据,是“空的”;而 void 指针实实在在地指向一块内存,只是不知道这块内存中是什么类型的数据。