前言
作者简介:热爱跑步的恒川,致力于C/C++、Java、Python等多编程语言,热爱跑步,喜爱音乐的一位博主。
本文收录于C语言进阶系列,本专栏主要内容为数据的存储、指针的进阶、字符串和内存函数的介绍、自定义类型结构、动态内存管理、文件操作等,持续更新!
相关专栏Python,Java等正在发展,拭目以待!
动态内存管理
- 题目1:
- 题目2:
- 题目3:
- 题目4:
题目1:
void GetMemory(char* p){p = (char*)malloc(100);}void Test(void){char* str = NULL;GetMemory(str);strcpy(str, "hello world");printf(str);}int main(){Test();return 0;}
请问运行Test 函数会有什么样的结果?
图片讲解:
p是形参,已离开就会被自动销毁,所以此时p不见了
该代码原理讲解:
- 调用GetMemory函数的时候,str的传参为值传递,p是str的临时拷贝,所以在GetMemory函数的内部将动态开辟空间的地址存放在p中的时候,不会影响str.所以GetMemory函数返回之后,str中依然是NULL指针,strcpy函数就会调用失败,原因是对NULL的解引用操作,程序会崩溃。
- GetMemory函数内容malloc申请的空间没有机会释放,造成了内存泄漏
题目2:
char* GetMemory(void){char p[] = "hello world";return p;}void Test(void){char* str = NULL;str = GetMemory();printf(str);}int main(){Test();return 0;}
请问运行Test 函数会有什么样的结果?
图片讲解:
该代码原理讲解:
返回栈空间地址的问题
GetMemory函数内部创建的数组是临时的,虽然返回了数组的起始地址给了str,但是数组的内存出了GetMemory函数就被回收了,而str依然保存了数组的起始地址,这时如果使用str,str就是野指针。
int* test(){int a = 10;return &a;}int main(){int* p = test();printf("hehe\n");printf("%d\n", *p);return 0;}
思考一下该代码的结果是什么?
答案是5,跟上面讲的原理一样
图片讲解:
题目3:
void GetMemory(char** p, int num){*p = (char*)malloc(num);}void Test(void){char* str = NULL;GetMemory(&str, 100);strcpy(str, "hello");printf(str);}int main(){Test();return 0;}
请问运行Test 函数会有什么样的结果?
答案是缺少释放
正确更改代码的样子:
void GetMemory(char** p, int num){*p = (char*)malloc(num);}void Test(void){char* str = NULL;GetMemory(&str, 100);strcpy(str, "hello");printf(str);//释放free(str);}int main(){Test();return 0;}
题目4:
void Test(void){char* str = (char*)malloc(100);strcpy(str, "hello");free(str);if (str != NULL){strcpy(str, "world");printf(str);}}int main(){Test();return 0;}
请问运行Test 函数会有什么样的结果?
答案是str没有被置为空指针
这才是正确的形式
void Test(void){char* str = (char*)malloc(100);strcpy(str, "hello");free(str);str = NULL;if (str != NULL){strcpy(str, "world");printf(str);}}int main(){Test();return 0;}
如果这份博客对大家有帮助,希望各位给恒川一个免费的点赞作为鼓励,并评论收藏一下⭐,谢谢大家!!!
制作不易,如果大家有什么疑问或给恒川的意见,欢迎评论区留言。