【C语言进阶篇】动态内存分配的六个常见错误

C/C++
188
0
0
2024-02-25

⛺️生活的理想,就是为了理想的生活!

文章目录
  • 前言
  • 💬 常见的动态内存错误
  • 1️⃣ 对NULL指针的解引用操作
  • ⌨️ 错误原因
  • 💻 解决方法:
  • 2️⃣ 对动态开辟空间的越界访问
  • ⌨️ 错误原因
  • 💻 解决方法:
  • 3️⃣ 对非动态开辟内存使用free释放
  • ⌨️ 错误原因
  • 4️⃣ 使用free释放一块动态开辟内存的一部分
  • ⌨️ 错误原因
  • 5️⃣ 对同一块动态内存多次释放
  • ⌨️ 错误原因
  • 💻 解决方法:
  • 6️⃣ 动态开辟内存忘记释放(内存泄漏)
  • ⌨️ 错误原因
  • 💻 内存泄漏危害:
  • 总结

前言

🌈hello! 各位宝子们大家好啊,前面一章给大家带来了动态内存分配的四个函数的讲解!malloc free calloc realloc 这四个库函数不知道大家还记得吗? ⛳️动态内存分配学完了那么,这些关于内存分配的错误你就必须避免了!

💬 常见的动态内存错误

1️⃣ 对NULL指针的解引用操作

📚 代码演示:

void test()
{
	int* p = (int*)malloc(INT_MAX / 4);
	*p = 20;//如果p的值是NULL,就会有问题
	free(p);
}
⌨️ 错误原因
⛳️ 这里我们对malloc的返回值用p接收,然后解引用进行赋值错误点在:
  • malloc 我们上一篇我们说了它的返回值有可能是空指针
  • 而我们对空指针进行解引用,会导致导致程序异常终止或拒绝服务
  • 所以这种肯定是错误的,所以一定要进行判断在进行接收

在这里插入图片描述

💻 解决方法:
void test()
{
	int* p = (int*)malloc(INT_MAX / 4);
	if (p == NULL)
	{
		perror("calloc");
		return 1;
	}
	*p = 20;//如果p的值是NULL,就会有问题
	free(p);
}

2️⃣ 对动态开辟空间的越界访问

📚 代码演示:

#include <stdio.h>
#include <stdlib.h>
int  main()
{
	int i = 0;
	int* p = (int*)malloc(10 * sizeof(int));
	if (NULL == p)
	{
		perror("malloc");
			return 1;
	}
	for (i = 0; i < 11; i++)
	{
		p[i] = i;//当i是10的时候越界访问
	}
	free(p);
	return 0;
}
⌨️ 错误原因
⛳️不知道大家看出来了没有,我们这里总共就申请了10个整形的空间。而调用的时候缺想访问11个整形进行赋值.
  • 这不就造成了越界访问,只申请了10个整形大小
  • 你却想用11个整形大小,谁给你的胆子 ⁉️😄
  • 这样就会照成程序的崩溃,所以一定要避免

📑 代码结果:

在这里插入图片描述

💻 解决方法:
#include <stdio.h>
#include <stdlib.h>
int  main()
{
	int i = 0;
	int* p = (int*)malloc(10 * sizeof(int));
	if (NULL == p)
	{
		perror("malloc");
			return 1;
	}
	for (i = 0; i < 10; i++)
	{
		p[i] = i;//当i是10的时候越界访问
	}
	free(p);
	return 0;
}
📌 所以解决办法也非常简单只要,我们避免就可以啦!

3️⃣ 对非动态开辟内存使用free释放

📚 代码演示:

#include <stdio.h>
#include <stdlib.h>
int  main()
{
	int a = 10;
	int* p = &a;
	free(p);//ok?
	return 0;
}
⌨️ 错误原因
⛳️free这个函数只能释放我们动态内存开辟的空间,结果你非要去栈区上释放空间
  • 这种代码一般是我们喝醉了才会这样用😂😂😂
  • 开个玩笑哈!
  • 注:关于free函数和栈区我们在动态函数时有详细讲解!《动态内存详细讲解》

📑 代码结果:

在这里插入图片描述

4️⃣ 使用free释放一块动态开辟内存的一部分

📚 代码演示:

#include <stdio.h>
#include <stdlib.h>
int  main()
{
	int* p = (int*)malloc(100);
	p++;
	free(p);//p不再指向动态内存的起始位置
	return 0;
}
⌨️ 错误原因
这种情况是释放了对于属于我们开辟的内存空间,没有释放完全
  • 当我们把p指针偏移之后,就不在指向原来的起始位置
  • 所以当我们对p指向的空间进行释放的时候就会进行 内存泄漏
  • 这是非常严重的编程错误,直接会导致程序崩溃

在这里插入图片描述

📑 代码结果:

在这里插入图片描述

5️⃣ 对同一块动态内存多次释放

📚 代码演示:

#include <stdio.h>
#include <stdlib.h>
int  main()
{
	int* p = (int*)malloc(100);
	free(p);
	free(p);//重复释放
	return 0;
}
⌨️ 错误原因
⛳️ 这也是写代码糊涂了才会这样,对一块内存进行多块释放!本来p的地址我们都已经释放完了你还让我释放一遍这怎么能这样呢?

📑 代码结果:

在这里插入图片描述

💻 解决方法:
这个解决方法也非常简单,我们只要每次 free 释放完把 p 置为空指针就可以了。
  • p 里面都是空指针,都没有存地址
  • 那么我们不管怎么释放都没问题了

在这里插入图片描述

6️⃣ 动态开辟内存忘记释放(内存泄漏)

📚 代码演示:

#include <stdio.h>
#include <stdlib.h>
void test()
{
	int* p = (int*)malloc(100);
	if (NULL != p)
	{
		*p = 20;
	}
}
int main()
{
	test();
	while (1);
	return 0;
}
;
⌨️ 错误原因
⛳️ 这个情况就有点像我们看的侦查片里面的卧底,我们派了 小王 进行卧底但是只有警员 阿SIR 记得小王的档案为了保密其他人都不知道,有一天 阿SIR 突然去世了,这时不就没人记得 小王的卧底身份了!
  • 这里我们test进行了申请空间但是没有进行释放
  • 而出了这个函数 p 也就进行销毁了没人记得这
  • 个开辟空间的地址
💻 内存泄漏危害:
⛳️那么内存泄漏有什么危害呢?这个其实很常见 内存病毒 不知道大家听说过没有?每次开机都没问题,但总是用不到多长时间就会死机!
  • 这个是不是就和内存泄漏很像,如果我们
  • 写了一个程序每次都申请空间都不释放
  • 那么时间长了这个程序就会直接崩溃! ⁉️

所以一定要避免这种情况,免得以后费力吧唧的写了一个代码人家说什么鬼程序用一下就崩! 🔥 注:动态开辟的空间一定要释放,并且正确释放 。

总结

✅ 归纳: 好了以上就是关于动态内存分配的全部注意事项就全部讲解完毕啦! 对NULL指针的解引用操作对动态开辟空间的越界访问对非动态开辟内存使用free释放使用free释放一块动态开辟内存的一部分对同一块动态内存多次释放 ☁️ 好了这些就是常见动态内存分配的错误了,大家一定要避免啊!