当前位置:网站首页>C - 为什么指针常常初始化为 NULL?
C - 为什么指针常常初始化为 NULL?
2022-08-03 10:57:00 【放羊的牧码】
关于空指针NULL、野指针、通用指针,首先说一下什么是指针,只要明白了指针的含义,你就明白null的含义了
假设有语句
int a = 123;
那么编译器就在内存中开辟 1 个整型单元存放变量 a,我们假设这个整型单元在内存中的地址是 0x100;那么内存 0x100 单元中存放了数据 123,每次我们访问a的时候,实际上都是访问的0x100单元中的 123
现在定义
int *p;
p = &a;
当编译器遇到语句 int *p 时,它也会在内存中给指针变量 p 分配一个内存单元,假设这个单元在内存的编址为 0x200;此时 0x200 中的值是不确定的(因为我们没有给指针赋值),当编译器遇到了p = &a 时,就会在 0x200 单元中保存 0x100,这就是说指针变量 p 代表的内存单元 0x200 存放了变量 a 的内存地址!用通俗的话说就是 p 指向了变量 a。
p=NULL,就是说:内存单元 0x200 不存放任何变量的内存地址。
删除一个 new 了的数组。有必要的话。比如非标准的类(new MyClass),在 Type *p = new Type[N]; delete []p;的最后最好再加一句:p = NULL
空指针是一个特殊的指针值,也是唯一一个对任何指针类型都合法的指针值。指针变量具有空指针值,表示它当时处于闲置状态,没有指向有意义的东西。空指针用 0 表示,C语言保证这个值不会是任何对象的地址。给指针值赋零则使它不再指向任何有意义的东西。为了提高程序的可读性,标准库定义了一个与 0 等价的符号常量NULL。程序里可以写 p = 0; 或者 p = NULL; 两种写法都把p置为空指针值。相对而言,前一种写法更容易使读程序的人意识到这里是一个指针赋值。
我们印象中C语言的指针都有类型,实际上也存在一种例外。这里涉及到通用指针,它可以指向任何类型的变量。通用指针的类型用(void *)表示,因此也称为void 指针。
int n=3, *p;
void *gp;
gp = &n;
p=(int *)gp;
野指针,也就是指向不可用内存区域的指针。通常对这种指针进行操作的话,将会使程序发生不可预知的错误。
“野指针”不是NULL指针,是指向“垃圾”内存的指针。人们一般不会错用NULL指针,因为用 if 语句很容易判断。但是“野指针”是很危险的,if语句对它不起作用。
- 野指针的成因主要有两种
- 指针变量没有被初始化。任何指针变量刚被创建时不会自动成为NULL指针,它的缺省值是随机的,它会乱指一气。所以,指针变量在创建的同时应当被初始化,要么将指针设置为NULL,要么让它指向合法的内存
- 指针p被free或者delete之后,没有置为NULL,让人误以为p是个合法的指针。别看free和delete的名字恶狠狠的(尤其是delete),它们只是把指针所指的内存给释放掉,但并没有把指针本身干掉。通常会用语句if (p != NULL)进行防错处理。很遗憾,此时if语句起不到防错作用,因为即便p不是NULL指针,它也不指向合法的内存块。例
char *p = (char *) malloc(100);
strcpy(p, “hello”);
free(p); // p 所指的内存被释放,但是p所指的地址仍然不变
if(p != NULL) // 没有起到防错作用
strcpy(p, “world”); // 出错
另外一个要注意的问题:不要返回指向栈内存的指针或引用,因为栈内存在函数结束时会被释放。
指针是个很强大的工具,可是正因为它太强大,所以要操作它不是件易事。操作不当造成的野指针,甚至会引起系统死机等比较严重的后果。
如果程序定义了一个指针,就必须要立即让它指向一个我们设定的空间或者把它设为NULL,如果没有这么做,那么这个指针里的内容是不可预知的,即不知道它指向内存中的哪个空间(即野指针),它有可能指向的是一个空白的内存区域,可能指向的是已经受保护的区域,甚至可能指向系统的关键内存,如果是那样就糟了,也许我们后面不小心对指针进行操作就有可能让系统出现紊乱,死机了。所以我们必须设定一个空间让指针指向它,或者把指针设为NULL,这是怎么样的一个原理呢,如果是建立一个与指针相同类型的空间,实际上是在内存中的空白区域中开辟了这么一个受保护的内存空间,然后用指针来指向它,那么指针里的地址就是这个受保护空间的地址了,而不是不可预知的啦,然后我们就可以通过指针对这个空间进行相应的操作了;如果我们把指针设为NULL,我们在头文件定义中的 #define NULL 0 可以知道,其实NULL就是表示0,那么我们让指针=NULL,实际上就是让指针=0,如此,指针里的地址(机器数)就被初始化为0了,而内存中地址为0 的内存空间……不用多说也能想象吧,这个地址是特定的,那么也就不是不可预知的在内存中乱指一气的野指针了。
还应该注意的是,free和delete只是把指针所指的内存给释放掉,但并没有把指针本身干掉。指针p被free以后其地址仍然不变(非NULL),只是该地址对应的内存是垃圾,p成了“野指针”。如果此时不把p设置为NULL,会让人误以为p是个合法的指针。用free或delete释放了内存之后,就应立即将指针设置为NULL,防止产生“野指针”。内存被释放了,并不表示指针会消亡或者成了NULL指针(而且指针消亡了,也并不表示它所指的内存会被自动释放)
最后,总结一下野指针的的成因吧
1、指针变量没有被初始化。任何指针变量刚被创建时不会自动成为NULL指针,它的默认值是随机的,它会乱指一气
2、指针p被free或者delete之后,没有置为NULL,让人误以为p是个合法的指针
3、指针操作超越了变量的作用范围。这种情况让人防不胜防
边栏推荐
- What is the ERC20 token standard?
- complete knapsack problem
- 巴比特 | 元宇宙每日必读:玩家离场,平台关停,数字藏品市场正逐渐降温,行业的未来究竟在哪里?...
- How to retrieve IDC research reports?
- 白帽黑客与留守儿童破壁对“画”!ISC、中国光华科技基金会、光明网携手启动数字安全元宇宙公益展
- 再谈“雷克萨斯”安全装置失效!安全手册疑点重重,网友:细思极恐
- 程序员架构修炼之道:软件架构基本概念和思维
- 鸿蒙第三次
- 【输出一个整数的的每一位,由高到低输出。使用递归和不使用递归】
- 【AppCube】数字孪生万物可视 | 联接现实世界与数字空间
猜你喜欢
C#/VB.NET 从PDF中提取表格
The way of programmer architecture practice: how to design a sustainable evolution system architecture?
如何检索IDC研究报告?
如何改变sys_guid() 返回值类型
[LeetCode—Question 2 Sum of Two Numbers Detailed Code Explanation ] The source code is attached, which can be copied directly
Skills required to be a good architect: How to draw a system architecture that everyone will love?What's the secret?Come and open this article to see it!...
go——并发编程
使用.NET简单实现一个Redis的高性能克隆版(一)
【LeetCode—第2题 两数之和 代码详解 】附有源码,可直接复制
怎么在外头使用容器里php命令
随机推荐
Depth study of 100 cases - convolution neural network (CNN) to realize the clothing image classification
ARIMA实现(亲测可用)
记某社区问答
Classical Architecture and Memory Classification of Embedded Software Components
QT with OpenGL(Shadow Mapping)(面光源篇)
巴比特 | 元宇宙每日必读:玩家离场,平台关停,数字藏品市场正逐渐降温,行业的未来究竟在哪里?...
如何通过DBeaver 连接 TDengine?
gbase在轨道交通一般都采用哪种高可用架构?
go——并发编程
程序员架构修炼之道:如何设计出可持续演进的系统架构?
STM32+OLED显示屏制作指针式电子钟
507. 完美数
创建C UDR时,指定的HANDLESNULLS的作用是什么?
QT with OpenGL(HDR)
谷歌实用插件分享
【JDBC以及内部类的讲解】
数字藏品和ICP
跨链桥协议 Nomad 遭遇黑客攻击,损失超 1.5 亿美元
机器比人更需要通证
Skills required to be a good architect: How to draw a system architecture that everyone will love?What's the secret?Come and open this article to see it!...