解析令人纠结的getMemory()函数:多种情况下的详细分析与解决方案探讨
csdn 2023-12-05

如果你将面试一份C/C++的工作,那么无论是笔试题或者面试题都有极大可能会被问到getMemory()的问题。当然这也是一道比较纠结的题目,本文就对这几道题目来做一个分析对比。

题目一

void getMemory(char *p) { p=(char *)malloc(100); } void Test(void) { char *str=NULL; getMemory(str); strcpy(str,"hello world"); printf(str); } 

运行结果:运行错误
解释:getMemory(char *p)中的函数参数是char *类型的,而传入函数的str的类型也是char *。这就导致了,调用getMemory(str);后,最终str的值还是NULL,在函数内部修改形参并不能真正改变传递进去的实参。简单一点说,要想改变一级指针,需要传递一级指针的地址,也就是二级指针。

题目二

char *getMemory(void) { char p[]="hello world"; return p; } void Test(void) { char *str=NULL; str=getMemory(); printf(str); } 

运行结果:运行无误,但打印乱码
解释:getMemory(void)中的p[]为函数内的局部自动变量,在函数返回后,内存已经被释放。如果一步步调试,会发现执行str=getMenory();后str不再是NULL了,但是str的内容并不是hello world,而是垃圾数据。

题目三

void getMemory(char **p,int num) { *p=(char *)malloc(num); } void Test(void) { char *str=NULL; getMemory(&str,100); strcpy(str,"hello world"); printf(str); } 

运行结果:运行正确,但有内存泄漏
解释:getMemory(char **p,int num)中的中的函数参数是char **p类型的,而传入函数的str的类型是char *。利用二级指针修改一级指针,没有问题。但是动态分配的内存并不会自动释放,容易有内存泄漏的风险。同时,没有测试是否成功分配了内存,应该有if(*p==NULL) { ……}之类的语句处理内存分配失败的其情况。

题目四

char* getMemory(int num) { char* p=(char*)malloc(num); return p; } void Test(void) { char* str=NULL; str=getMemory(100); strcpy(str,"hello world"); printf(str); } 

运行结果:运行正确,但有内存泄漏
解释:注意题目五和题目二的区别。虽然都是局部变量,但题目五用函数返回值来传递动态内存的地址;而题目二return语句返回指向“栈”内存的指针,因为该内存在函数结束时自动消亡。同样,动态分配的内存并不会自动释放,容易有内存泄漏的风险。

题目五

char* getMemory(void) { char* p="hello world"; return p; } void Test(void) { char* str=NULL; str=getMemory(); printf(str); } 

运行结果:运行正确,但不合理
解释:getMemory(void)中的中,p指向的是字符串常量,字符串常量保存在静态存储区。虽然Test(void)运行不会出错,但是函数getMemory(void)的设计概念却是错误的。因为getMemory(void)内的“hello world”是常量字符串,位于静态存储区,它在程序生命期内恒定不变。无论什么时候调用getMemory(void),它返回的始终是同一个“只读”的内存块。例如,如想执行

p[0]='n'; 

则程序会中断,并提示内存错误。

主要区分

1、栈中分配局部变量空间,是系统自动分配空间。由于栈上的空间是自动分配自动回收的,所以栈上的数据的生存周期只是在函数的运行过程中,运行后就释放掉,不可以再访问;
2、堆区分配程序员申请的内存空间,堆上的数据只要程序员不释放空间,就一直可以访问到,不过缺点是一旦忘记释放会造成内存泄露;
3、静态区是分配静态变量,全局变量空间的。

声明: 本文转载自其它媒体或授权刊载,目的在于信息传递,并不代表本站赞同其观点和对其真实性负责,如有新闻稿件和图片作品的内容、版权以及其它问题的,请联系我们及时删除。(联系我们,邮箱:evan.li@aspencore.com )
0
评论
  • 【7.24 深圳】2025国际AI+IoT生态发展大会/2025全球 MCU及嵌入式技术论坛


  • 相关技术文库
  • C语言
  • 编程
  • 软件开发
  • 程序
  • 51单片机数码管显示跑马灯程序源代码讲解

    基于51单片机学习板。用S1键作为控制跑马灯的方向按键,S5键作为控制跑马灯方向的加速度按键,S9键作为控制跑马灯方向的减速度按键,S13键作为控制跑马灯方向的启动或者暂停按键。记得把输出线P0.4一直输出低电平,...

    昨天
  • 基于ARM的智能家居控制通信控制站的设计与实现

    0 引言 物体信息化是现代社会信息化建设的“催化剂”和“增倍器”。只有走集成整合信息技术以及信息数据之路,企业的信息化建设才能真正发挥作用,才能进一步推动信息建设上水平。现代物体信息化的发展,直接刺激了新生...

    前天
  • ARM中打印函数print 的几种实现方法

    1利用C库函数printf。 步骤: 1)首先需要包含头文件stdio.h。 2)然后定义文件句柄。实际上就是一个int型变量封装在结构体中。 struct__FILE{inthandle;}; 3)定义FILE__stdout;FILE即为__FILE,通过stdio.h宏定义。...

    前天
  • 高效的C编程之:C编译器及其优化(上)

    本章将帮助读者在ARM处理器上编写高效的C代码。本章涉及的一些技术不仅适用于ARM处理器,也适用于其他RISC处理器。本章首先从ARM编译器及其优化入手,讲解C编译器在优化代码时所碰到的一些问题。理解这些问题,将有...

    07-08
  • 有关C51的编程规范

    简介:编程首要是要考虑程序的可行性,然后是可读性、可移植性、健壮性以及可测试性。这是总则。但是很多人忽略了可读性、可移植性和健壮性(可调试的方法可能歌不相同),这是不对的。 下面就来说说有关C51的编程规...

    07-08
  • 光立方程序编写步骤

    基于51单片机的4*4*4光立方程序实现原理及程序代码。LED光立方的复位电路、时钟电路、每层LED灯电路控制逻辑,系统总原理图,工作流程及相关C语言源码实现。希望能够对你学习了解LED光立方程序编写及LED立方实体制...

    07-04
  • 封装继承多态

    封装: 封装是实现面向对象程序设计的第一步,封装就是将数据或函数等集合在一个个的单元中(我们称之为类)。被封装的对象通常被称为抽象数据类型。 封装的意义: 封装的意义在于保护或者防止代码(数据)被我们无意中...

    07-04
  • 封装是什么意思?

    即隐藏对象的属性和实现细节,仅对外公开接口,控制在程序中属性的读和修改的访问级别;将抽象得到的数据和行为(或功能)相结合,形成一个有机的整体,也就是将数据与操作数据的源代码进行有机的结合,形成“类”,其中...

    07-04
  • 超声波模块测距51程序_单片机超声波测距c语言

    超声波检测原理 超声波测距的程序流程图 程序如下: //超声波模块程序 //超声波模块程序 //Trig = P2^0 //Echo = P3^2 #include #define uchar unsigned char #define uint unsigned int // void delay(uint z) {...

    07-01
  • 大佬带你看嵌入式系统,嵌入式系统该学习什么?

    嵌入式系统是当今的热门系统之一,在诸多领域,嵌入式系统都有所应用。为增进大家对嵌入式系统的认识,小编将为大家介绍嵌入式系统是一个什么样的专业,以及学习嵌入式系统该学习哪些内容。如果你对嵌入式系统具有...

    06-27
  • c51单片机编程要点总结

    c51单片机编程要点总结 1、头文件:#include (我用的是 STC 89C54RD+) 2、预定义:sbit LED = P1^0// 定义 P1 口的 0 位为 LED 注:“P1^0”这个写法,与 A51 不同(A51 是 P1.0),P1 是一组端口,端口号范围 0~7 注2...

    06-25
下载排行榜
更多
评测报告
更多
广告