微波EDA网,见证研发工程师的成长!
首页 > 研发问答 > 嵌入式设计讨论 > MCU和单片机设计讨论 > 细思则恐!C语言里NULL从本质上来理解

细思则恐!C语言里NULL从本质上来理解

时间:10-02 整理:3721RD 点击:
细思则恐!C语言里NULL从本质上来理解
NULL一看名字就知道为空,什么为空呢,而且全是大写。是不是意味着是个宏定义呢?如果想到这里,我相信你离真理不远了。
有些人为什么犯错?因为只看到了NULL这四个字母而已,没有看到事物的本质,老师在课堂上也只是说空指针,空字符串。这样只会陷入无止境的误区。如果我这样定义:
#define NULL 0
你是不是又该纠结了呢?
正确的做法是把它当作一个宏,不管如何变化,来展开看看就知道了。
在C语言的头文件stddef.h中,NULL的定义如下:
#define NULL
#define __cplusplus //如果定义了__cplusplus表示是c++程序
#define NULL 0
#else
#define NULL ((void *)0)
#endif
#endif
#interface NULL 0
所以,c语言里NULL的本质是(void *)0。
这里一开始就犯了个错,强制转换的意义是什么?感谢朱老师物联网大课堂的朱老师以及群里的朋友们的激烈讨论。以下内容截取自我们的讨论。
((void *)0) 本质上是将数字0强制转换为(void *)。
0其实就是地址,(void *)只是说我们认为0这个地址中存储的类型是void *,也就是说0这个地址中存储的类型是void的,也就是当前不知道还未指定的。
0 地址是虚拟地址 还是物理地址?
不管是虚拟还是物理地址,就是当前环境的0地址。
要理解强制类型转换的本质
0还是0,永远是0,表示的永远是0这个内存地址;前面的类型,只是告诉我们我们认为0这个地址中装了什么
首先 这里的数字0 就是表示地址0?
不能讲某个数字直接强制转化为指针吗
举个例子就可以了:
#define GPJ0CON (unsigned int *)0x20008000
如果是需要将一个数字强制转换为指针,是不是可以这样写呢 int * &2, 是不是强制讲2转化为指针类型了呢
这样啊 (unsigned int *)0x2
char i=2; 间接讲2转换为int 指针。是不是这样写int * &i
char i = 2;之后 编译器会给i变量分配一个地址,&i 就是取i变量的地址。变量定义后就会分配地址。
我说的0,你这里写的2本身这个数字就是表示地址,而你的理解是2是值,这个值存在一个变量中,所以想通过&2去取2的地址,这个是错误的。因为只有变量才有地址,2只是个数字是没有地址的。
而单纯的数字本身就是地址。
是的,在这里确实是这样,因为这个数字我们要把它强制类型转化成指针,所以这个数字就是地址
C语言里有很多东西都是这种,规则只是讲了法律,但是实际应用中有很少常用技巧、或者衍生出来的用法,都没有提及,这也是我们的C高级课程中需要补充的。
因为只有变量才有地址,2只是个数字是没有地址的.因为这个数字我们要把它强制类型转化成指针,所以这个数字就是地址
C语言中就通过类型来表示这个单元中存的什么玩意,譬如(int)0就表示0地址中存的是int型的数;譬如你(int *)0就表示0地址中存的是个地址,这个地址指向的单元是int型数
NULL只是一个概念,叫作空值,其值本身没有任何含义,可以用0代替,也可以用1,...代替,只要这些值不会与系统实际的有效地址冲突即可。
因此,本人在此再次强调,不要自作聪明地认为NULL就是0,要判断的时候还是老老实实地与NULL做比较,别想当然地用什么!ptr之类的写法,因为在某个特定环境下,NULL可能不是0,而系统函数返回的是NULL不是0,那时,你的函数就会出现莫名其妙的错误。所以,养成良好的习惯是非常重要的。
从这里学到了一点 如果看到了一个宏不会用 那么最好的办法 不是去看别人的解释 而是展开它,看它的最终展开形式。
总结如下:
C语言中NULL表示内存位置0,NULL((void *)0)指针并不指向任何对象。因此除非是用于赋值或比较运算,出于其他任何目的使用NULL指针都是非法的。
注意:在ASCII码的第一个字符是NULL,它的数值是0,占用1字节;c语言把它作为字符串的最后一个字符,以表示字符串到此结束."\0"
接下来:总结下NULL具体用在哪些场合。
第一:指针初始化(对应于上面的赋值)。
例如 int *x = NULL;
第二:函数返回是否成功判断(对应于做比较运算)。
当函数返回值为指针时,判断函数是否返回成功。举例如下:
FILE * pFile;
pFile = fopen ("myfile.txt" , "r");
if (pFile == NULL) printf("open error");

第三:判断字符串是否为空字符串(对应于做比较运算)。
if(str==NULL)或者if(str[0]=='\0') 就是空

以下课程可免费试听C语言、电子、PCB、STM32、Linux、FPGA、JAVA、安卓等。
想学习的你和我联系预约就可以免费听课了。
宋工企鹅号:3524-6590-88   Tel/WX:173--1795--1908


学习了!学习了!学习了!学习了!学习了!学习了!

感谢作者的文章,终于搞明白为什么有时候报错了。

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top