今天遇见了一道有意思的内存泄露题目,特地分享给大家,相信屏幕前的你学习完一定有所收获,预祝读者学习愉快,多多进步早日升职加薪
一、例题
试问该段代码能打印什么,或者不能打印什么,说出理由
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
void getmemory(char*p)
{
p = (char*)malloc(100);
}
void test()
{
char*str = NULL;
getmemory(str);
strcpy(str, "hello world");
printf("%s",str);
}
int main()
{
test();
return 0;
}
这段代码乍一看好像没有啥问题,test函数进去创建一个str嘛,然后str传给getmemory函数,开辟一块空间,然后hello world这字符串放进开辟的空间里,然后打印str也就是hello world嘛。
但是,上面的思路是完全错误的,代码完全无法运行,解释如下:
- 第一:你str传到getmemory函数里,p开辟一个空间,被开辟的空间和str没有关系
(传值调用和传址调用的区别)
- 第二:你getmemory函数出来之后,开辟的空间没有释放掉,而且你p由于函数的结束已经销毁了,p销毁之后再也无法找到开辟的空间,造成内存泄漏
就比如p是一个警察头子,开辟的空间是卧底,只有p知道那个卧底,你现在警察头子死了,再也没法证明卧底是卧底了
也就是说p消失后,没办法再对开辟的空间进行释放,这时就会造成内存泄露
二、2种改进方法
法1:二级指针(传址调用)
代码如下(示例):
void getmemory(char**p)
{
*p = (char*)malloc(100);//p是str地址,*p是str
}
void test()
{
char*str = NULL;
getmemory(&str);
strcpy(str, "hello world");
printf("%s",str);
free(str);
str = NULL;
}
int main()
{
test();
return 0;
}
因为之前的传值调用没法让p和str建立联系,那我们就传地址,开辟一块空间出来赋给*p
要注意的是,p是str地址嘛,*p也就是str,到这里,str掌握了新开辟空间的控制权。然后getmemory函数出来,进行strcpy和printf操作,最后不要忘记free掉原先的空间。
法2:返回指针
代码如下(示例):
char* getmemory(char*p)
{
p = (char*)malloc(100);
return p;
}
void test()
{
char*str = NULL;
str=getmemory(str);
strcpy(str, "hello world");
printf("%s", str);
free(str);
str = NULL;
}
int main()
{
test();
return 0;
}
因为这里是开辟了一块空间给指针p管理嘛,str和p没有建立联系,那你直接函数返回p这个指针然后赋值给str即可,随后操作和方法1一样
总结
本文介绍了动态内存分配的常见例题及两种改进方法,读者需要理解为何会发生内存泄露和传值与传址调用的区别就可以熟练解决这类问题。以上,祝读者学习愉快~希望大家以后多多支持编程学习网!
本文标题为:C语言动态内存泄露常见问题内存分配改进方法详解
基础教程推荐
- 一文带你了解C++中的字符替换方法 2023-07-20
- C++使用easyX库实现三星环绕效果流程详解 2023-06-26
- C++详细实现完整图书管理功能 2023-04-04
- 详解c# Emit技术 2023-03-25
- 如何C++使用模板特化功能 2023-03-05
- C利用语言实现数据结构之队列 2022-11-22
- C++中的atoi 函数简介 2023-01-05
- C语言基础全局变量与局部变量教程详解 2022-12-31
- C语言 structural body结构体详解用法 2022-12-06
- C/C++编程中const的使用详解 2023-03-26