关于内存的思考题(摘自林锐高质量c++编程)

King_Dragon 2005-09-22 11:42:43
下面两个内存使用的例子中,存在一定相似:使用malloc分配内存。运行结果截然不同,本人愚钝,不能理解通透,请各位大侠给出对原理的思考。
1.运行结果:程序崩溃,原因是GetMemory不能传递动态内存。
void GetMemory(char *p)
{
 p = (char *)malloc(100);
}
void Test(void)
{
 char *str = NULL;
 GetMemory(str);
 strcpy(str, "hello world");
 printf(str);
}
2.运行正常,内存泄漏
void GetMemory2(char **p, int num)
{
 *p = (char *)malloc(num);
}
void Test(void)
{
 char *str = NULL;
 GetMemory(&str, 100);
 strcpy(str, "hello");
 printf(str);
}
...全文
642 8 打赏 收藏 转发到动态 举报
写回复
用AI写文章
8 条回复
切换为时间正序
请发表友善的回复…
发表回复
HouJinkun2005 2005-09-23
  • 打赏
  • 举报
回复
这个问题首先是要理解程序里内存堆和内存栈的不同。
所有的函数调用,都利用了栈,参数/函数内的变量(局部变量)都放到了栈内,函数调用结束,出栈,里面的信息就全没了。
而堆,是动态分配内存空间的地方。所有的new在堆里,堆里的东西只有你去delete然整个程序结束后都不会清除,出现常说的内存泄露。

1.运行结果:程序崩溃,原因是GetMemory不能传递动态内存。
void GetMemory(char *p)
{
 p = (char *)malloc(100);
}
void Test(void)
{
 char *str = NULL;
 GetMemory(str);
 strcpy(str, "hello world");
 printf(str);
}
p是个形式参数,它的空间在栈里,函数结束它就没了。它在函数内有个值,该值是个地址,指向malloc出来的那块内存的地址。(那块内存在堆里)。
程序的问题很简单了,你在外部strcpy(str, "hello world"); 而str其实是Test里的一个局部变量,它只是作为实参传到那个函数里,它本身没有什么变化。它的值是NULL,所以你程序一执行就会程序崩溃。

2.运行正常,内存泄漏
void GetMemory2(char **p, int num)
{
 *p = (char *)malloc(num);
}
void Test(void)
{
 char *str = NULL;
 GetMemory(&str, 100);
 strcpy(str, "hello");
 printf(str);
}
这个程序,不同的是形式参数变成**p,这就造成了传地址现象。既然传的东西是地址,那么*p=(char *)malloc(num);当然会有用。(str的地址传给了**p啊)
但是这个程序没有释放堆,所以会出现内存泄露。
应该在printf(str);后面再加上free语句。

记得林锐的书里工讲了四个程序啊。两外两个和栈就有关系了。
ywchen2000 2005-09-23
  • 打赏
  • 举报
回复
都讲完了.
呵呵,学习
GrassJing526 2005-09-23
  • 打赏
  • 举报
回复
请大家在下面小节中,注意两个函数的参数形递有何不同,并思考各自的目的。

 

(结构变量以传值的方式传递) 

第一道:写一个函数,对所给的四方形求周长。

我们采用“传值”方式。传值将复制一份实参,然后把“复制品”传给函数。

//求周长:

int QiuZhouChang(SiFangXing sfx)

{

return sfx.l1 + sfx.l2 + sfx.w1 + sfx.w2; //四边之和即是周长

}

 

(结构变量以传址的方式传递) 

第二道:写一个函数,将所给的四方形长宽各增加一倍。

 

//加倍长宽

void JiaBeiChangKuan(SiFangXing& sfx)

{

//各边都*2;

sfx.l1 *= 2;

sfx.l2 *= 2;

sfx.w1 *= 2;

sfx.w2 *= 2;

}


看出区别了吗!!
emp 2005-09-22
  • 打赏
  • 举报
回复
楼上的兄弟都讲完了
vikingemail 2005-09-22
  • 打赏
  • 举报
回复
这个好像讨论过了。

第一个是传值,当函数执行完毕后没有传回来,你Test函数中的str仍然是NULL。

第二个是传址,你的str里面保存是申请到的地址,所以正常。

顺便说一下:
面试时好像这类的题比较的多。还是要弄清楚的好。
zhouhuahai 2005-09-22
  • 打赏
  • 举报
回复
void GetMemory(char *p)
{
 p = (char *)malloc(100);
}
在这里的传递方式就指针P本身而言,是传值的.也就是说类似于:
char *temp;
temp = p;
 temp = (char *)malloc(100);

所以在函数返回时,GetMemory(str);str并没有改变.它还是NULL.
而程序二中void GetMemory2(char **p, int num)
对*p而言,是传址的.
内存泄漏是因为malloc了的没有free掉
晨星 2005-09-22
  • 打赏
  • 举报
回复
除了引用之外,参数传递永远是传值,所以如果不使用引用同时又想对函数外部有所影响的话,通常是:
要修改变量,就传地变量的地址(指针);
要修改指针,就传地指针的地址(多级指针)。
实际上指针变量也是变量,所以第二句话只是第一句话的一个特例。
s_topman 2005-09-22
  • 打赏
  • 举报
回复
up。记住参数传递是值传就行了,指针也是。

33,311

社区成员

发帖
与我相关
我的任务
社区描述
C/C++ 新手乐园
社区管理员
  • 新手乐园社区
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧