C语言野指针、规避野指针、assert宏断言

目录

a.野指针成因

1.指针未初始化

2.指针越界访问

3.指针指向的空间释放

b.规避野指针

1.指针初始化

2.小心指针越界

3.指针变量不再使用时,及时置NULL,指针使用之前检查有效性

4.避免返回局部变量的地址

c.assert宏断言的使用


概念:野指针就是指针指向的位置是不可知的(随机的、不正确的、没有明确限制的)。一句话,任何指向非自己管理或者不想指向空间的指针都是野指针。

a.野指针成因

1.指针未初始化
cpp 复制代码
#include <stdio.h>
int main()
{
	int* p;//局部变量指针未初始化,默认为随机值
	*p = 20;
	return 0;
}

上述代码中,没有为 p 指针初始化,无法确定 p 所指向的空间是否合法,直接解应用赋值,错误。

2.指针越界访问
cpp 复制代码
#include <stdio.h>
int main()
{
	int arr[10] = { 0 };
	int* p = &arr[0];
	int i = 0;
	for (i = 0; i <= 11; i++)
	{
		//当指针指向的范围超出数组arr的范围时,p就是野指针
		*(p++) = i;
	}
	return 0;
}

上述代码中,数组 arr 有效元素个数只有10个 ,但指针 p 却访问到了第11个空间,此空间不是合法空间,错误。

3.指针指向的空间释放
cpp 复制代码
#include <stdio.h>

int* test()
{
	int n = 100;
	return &n;
}

int main()
{
	int* p = test();
	printf("%d\n", *p);
	return 0;
	
}

上述代码中,n 变量作为test()函数的局部变量,当test()函数调用结束,函数栈帧销毁后,n 变量空间被销毁系统回收),此时 p 即为指向非法空间的指针,错误。

b.规避野指针

1.指针初始化

如果明确知道指针指向哪里就直接赋值地址,如果不知道指针应该指向哪⾥,可以给指针赋值NULL。NULL 是C语言中定义的一个标识符常量,值是0,0 也是地址,这个地址是无法使用的,读写该地址会报错。

NULL的 C 定义:

cpp 复制代码
#include <stdio.h>
int main()
{
	int num = 10;
	int* p1 = &num;
	int* p2 = NULL;
	return 0;
}

p1 用于指向 num 变量明确,所以 p1 初始化直接赋值 num 变量地址,p2 也许后期使用但现在不知道该指向那个空间,于是给其初始化赋值NULL。

2.小心指针越界

一个程序向内存申请了哪些空间,通过指针也就只能访问哪些空间,不能超出范围访问,超出了就是越界访问。

比如访问数组时,数组长度用sizeof(arr)/ szieof(type) 来丈量,使用动态内存开辟时,使用变量数值来代替常量或者使用宏定义

3.指针变量不再使用时,及时置NULL,指针使用之前检查有效性
cpp 复制代码
int main()
{
	int arr[10] = { 1,2,3,4,5,67,7,8,9,10 };
	int* p = &arr[0];
	for (i = 0; i < 10; i++)
	{
		*(p++) = i;
	}
	//此时p已经越界了,可以把p置为NULL
	p = NULL;
	//下次使⽤的时候,判断p不为NULL的时候再使⽤
	//...
	p = &arr[0];//重新让p获得地址
	if (p != NULL) //判断
	{
		//...
	}
	return 0;
}
4.避免返回局部变量的地址

c.assert宏断言的使用

我们可以把野指针想象成野狗,野狗放任不管是非常危险的,所以我们可以找⼀棵树把野狗拴起来,就相对安全了,给指针变量及时赋值为NULL,其实就类似把野狗栓起来,就是把野指针暂时管理起来。不过野狗即使拴起来我们也要绕着走,不能去挑逗野狗,有点危险;对于指针也是,在使用之前,我们也要判断是否为NULL,看看是不是被拴起来起来的野狗,如果是,不能直接使用,如果不是,我们再去使用。

assert.h 头件定义了宏 assert() ,用于在运行时确保程序符合指定条件,如果不符合,就报

错终止运行。这个宏常常被称为"断言"。

上面代码在程序运行到这一行语句时,验证变量 p 是否等于 NULL 。如果确实不等于 NULL ,程序继续运行,否则就会终止运行,并且给出报错信息提示。

assert() 宏接受⼀个表达式作为参数。如果该表达式为真(返回值非零), assert() 不会产生
任何作用,程序继续运行。如果该表达式为假(返回值为零), assert() 就会报错,
在标准错误

流 stderr 中写入一条错误信息,显示没有通过的表达式,以及包含这个表达式的文件名和行号。

assert() 的使用对程序员是非常友好的,使用 assert() 有几个好处:它不仅能自动标识文件和出问题的行号,还有⼀种无需更改代码就能开启或关闭 assert() 的机制。如果已经确认程序没有问题,不需要再做断言,就在 #include <assert.h> 语句的前面,定义一个宏 NDEBUG 。

assert() 的缺点是,因为引人了额外的检查,增加了程序的运行时间。一般我们可以在 Debug 中使用,在 Release 版本中选择禁用 assert 就行,在 VS 这样的集成开发环境中,在 Release 版本中,直接就是优化掉了。这样在debug版本写有利于程序员排查问题,在 Release 版本不影响用户使用时程序的效率。

assert宏更详尽的信息:assert - C++ Reference (cplusplus.com)

相关推荐
黎雁·泠崖1 小时前
指针收官篇:sizeof/strlen + 指针运算笔试考点全梳理
c语言·开发语言
lingran__1 小时前
数据在内存中的存储详解(C语言拓展版)
c语言·开发语言
superman超哥1 小时前
仓颉语言中异常处理入门的深度剖析与工程实践
c语言·开发语言·c++·python·仓颉
德福危险2 小时前
C语言数据类型与变量 系统总结笔记
c语言·笔记·算法
若风的雨3 小时前
ARM Trusted Firmware 启动流程:从汇编到 C 语言的渐进式初始化
c语言·汇编·arm开发
疑惑的杰瑞3 小时前
【C】顺序结构
c语言·内存划分
小龙报3 小时前
【初阶数据结构】从 “数组升级” 到工程实现:动态顺序表实现框架的硬核拆解指南
c语言·数据结构·c++·算法·机器学习·信息与通信·visual studio
SELSL3 小时前
Linux文件属性及目录
linux·c语言·linux目录文件·linux文件属性、目录api·linux文件属性
秦苒&4 小时前
【C语言指针五】转移表、回调函数、qsort、qsort函数的模拟实现
c语言·开发语言·c#
旧梦吟4 小时前
脚本网页 C与汇编详解
c语言·css3·html5