//NO.1:
程序首先申请一个char类型的指针str,并把str指向NULL(即str里存的是NULL的地址,*str为NULL中的值为0),调用函数的过程
中做了如下动作:1申请一个char
类型的指针p,2把str的内容copy到了p里(这是参数传递过程中系统所做的),3为p指针申请了100个空间,4返回Test函数.最后程序把字符
串hello
world拷贝到str指向的内存空间里.到这里错误出现了!str的空间始终为NULL而并没有实际的空间.深刻理解函数调用的第2步,将不难发现问题
所在!

void GetMemory(char *p)

{

p = (char*)malloc(100);

}

void Test(void)

{

char *str = NULL;

GetMemory(str);

strcpy(str, "hello world");

printf(str);

}

//请问运行Test函数后会是什么样的结果?

//NO.2:
程序首先申请一个char类型的指针str,并把str指向NULL.调用函数的过程中做了如下动作:1申请一数组p[]并将其赋值为hello
world(数组的空间大小为12),2返回数组名p付给str指针(即返回了数组的首地址).那么这样就可以打印出字符串"hello
world"了么?当然是不能的!因为在函数调用的时候漏掉了最后一步.也就是在第2步return数组名后,函数调用还要进行一步操作,也就是释放内存
空间.当一个函数被调用结束后它会释放掉它里面所有的变量所占用的空间.所以数组空间被释放掉了,也就是说str所指向的内容将不确定是什么东西.

char *GetMemory(void)

{

char p[] = "hello world";

return p;

}

void Test(void)

{

char *str = NULL;

str = GetMemory();

printf(str);

}

//NO.3:问题同NO.1,正确答案为可以打印出hello.但内存泄漏了!

void GetMemory(char **p, int num)

{

*p = (char*)malloc(num);

}

void Test(void)

{

char *str = NULL;

GetMemory(&str, 100);

strcpy(str, "hello");

printf(str);

}

//NO.4:
申请空间,拷贝字符串,释放空间.前三步操作都没有任何问题.到if语句里的判断条件开始出错了,因为一个指针被释放之后其内容并不是NULL,而是一个
不确定的值.所以if语句永远都不能被执行.这也是著名的"野"指针问题.所以我们在编写程序释放一个指针之后一定要人为的将指针付成NULL.这样就会
避免出现"野"指针的出现.有人说"野"指针很可怕,会带来意想不到的错误.

void Test(void)

{

char *str = (char*)malloc(100);

strcpy(str, "hello");

free(str);

if (str != NULL)

{

strcpy(str, "world");

printf(str);

}

}

void GetMemory1(char *p)

{

p = (char *)malloc(100);

}

void Test1(void)

{

char *str = NULL;

GetMemory1(str);

strcpy(str, "hello world");

printf(str);

}

//str一直是空,程序崩溃

char *GetMemory2(void)

{

char p[] = "hello world";

return p;

}

void Test2(void)

{

char *str = NULL;

str = GetMemory2();

printf(str);

}

char *GetMemory3(void)

{

return "hello world";

}

void Test3(void)

{

char *str = NULL;

str = GetMemory3();

printf(str);

}

//Test3 中打印hello world,因为返回常量区,而且并没有被修改过。Test2中不一定能打印出hello world,因为指向的是栈。

void GetMemory4(char **p, int num)

{

*p = (char *)malloc(num);

}

void Test4(void)

{

char *str = NULL;

GetMemory3(&str, 100);

strcpy(str, "hello");

printf(str);

}

//内存没释放

void Test5(void)

{

char *str = (char *) malloc(100);

strcpy(str, "hello");

free(str);

if(str != NULL)

{

strcpy(str, "world");

printf(str);

}

}

//str为野指针,打印的结果不得而知

void Test6()

{

char *str=(char *)malloc(100);

strcpy(str, "hello");

str+=6;

free(str);

if(str!=NULL)

{

strcpy(str, "world");

printf(str);

}

}

//VC断言失败,运行错误

另转一则:

char *GetMemory3(int num)

{

char *p = (char *)malloc
(sizeof(char) * num);

return p; //返回指针
p

}
 
void Test3(void)

{

char *str = NULL;

str = GetMemory3(100);  //这里指针
str也和指针
p的指向一样的内存块。

strcpy(str, "hello");

cout<< str << endl;

free(str);  //这里最终只是释放了str指向的内存块,对于str指针
依然还是指向那块内存,看了一些资料说应该加一句str=0;那指针
p呢?p也依然在指向那块内存啊。

}

回复:

答者:wutaozhao() 信誉:100 级别:user2 日期:2007-6-29 21:42:21 id:41451636

指针
之间传递的就是地址,所以str和p所指向的内存块是同一个,释放其中任何一个即可

答者:what_a_big() 信誉:100 级别:user1 日期:2007-6-29 21:42:21 id:41451637

指针
p呢?
=============================================================================
after returning from GetMemory()
p (局部
变量)没了,那块内存还在。
现在str 指向那块内存。

看了一些资料说应该加一句str=0
=================================
也可以不写str=NULL,只要你不再引用str。

答者:believefym(feng) 信誉:100 级别:user5 日期:2007-6-29 21:50:38 id:41451675

同一块内存,释放一次即可,只不过所有指向那块内存的指针
都失效了,就像楼主代码里的p和str

释放之后str就变成了野指针
,加str=NULL,可以避免之后错误使用该指针
,方便调试

答者:lightnut() 信誉:100 级别:star1 日期:2007-6-29 22:05:48 id:41451736

1.
函数进入Test3():
char* str = NULL;

======================
变量   变量的地址         变量的值(内容)
str    0x0013fe8c         0

2.
str = GetMemory3(..)=====>
char *p = (char *)malloc
(sizeof(char) * num);
将分配的内存起始地址(0x003a60b0)赋给栈变量p,

===============================================
变量     变量的地址       变量的值(内容)
p        0x0013fda8       0x003a60b0

3. 从GetMemory3(..)返回, p的值(0x003a60b0)
拷贝给str, 栈变量p生命结束, 释放其所占栈内存
(地址0x0013fda8)

=================================================
变量       变量的地址     变量的值(内容)
str        0x0013fe8c      0x003a60b0

4. free(str):  释放str指向的内存(0x003a60b0开始的内存空间)
5. 推出Test3()后,  栈变量str生命结束, 释放其所占栈内存
(地址0x0013fe8c)

答者:freshui(五月的风 -最近老犯困) 信誉:100 级别:user1 日期:2007-6-29 22:12:07 id:41451763

呵呵 p没了
已经死了 
:)

注意看变量的作用域

答者:buhaohaoxuexi() 信誉:100 级别:user1 日期:2007-6-29 22:37:00 id:41451888

为什么会死呢,不是说动态分配是分配在堆上的吗?他不应该死啊。

答者:yydrewdrew(满堂花醉三千客,一剑霜寒十四州) 信誉:100 级别:user1 日期:2007-6-29 23:55:01 id:41452325

char *GetMemory3(int num)

{

char *p = (char *)malloc
(sizeof(char) * num);

return p; //返回指针
p

}
 
void Test3(void)

{

char *str = NULL;

str = GetMemory3(100);  //调用完后p被析构,p为局部
作用域存在于函数GetMemory3的栈中,当GetMemory3被调用完后,栈空间被释放,因而被析构。

strcpy(str, "hello");

cout<< str << endl;

free(str);  //这里最好加一句str = NULL。

}

答者:yydrewdrew(满堂花醉三千客,一剑霜寒十四州) 信誉:100 级别:user1 日期:2007-6-29 23:56:59 id:41452334

注意p和它指向的对象的生命期是两回事
p是一个局部
指针
,而它指向的对象是堆空间具有全局生命期

答者:xlbdan(流浪剑客) 信誉:100 级别:user5 日期:2007-6-30 0:15:43 id:41452409

return p; //返回指针
p

这里返回之后,p已经消失.
但是p指向的那块内存依然存在,并且被返回到了调用端,用str接收了.
所以只要free(str)就行了,也就释放了p申请的那块内存
然后把 str赋为NULL,是防止再错误的去使用它

getMemory的经典例子的更多相关文章

  1. python之路第五篇之递归(进阶篇:续:经典例子剖析)

    递归 在函数内部,可以调用其他函数; 如果一个函数在内部调用自身本身,这个函数就是递归函数. 例如,我们来计算阶乘: n! = 1 x 2 x 3 x ... x n, 用函数f1(n)表示,可以看出 ...

  2. jQuery监听事件经典例子

    关键字:jQuery监听事件经典例子  js代码:  ============================================================  $(function( ...

  3. linux Posix 信号量 三 (经典例子)

    本文将阐述一下信号量的作用及经典例子,当中包括“<越狱>寄信”,“家庭吃水果”,“五子棋”,“接力赛跑”,“读者写者”,“四方恋爱”等 首先,讲 semWait操作(P操作)和semSig ...

  4. sql优化经典例子

    场景 我用的数据库是mysql5.6,下面简单的介绍下场景 课程表 create table Course( c_id int PRIMARY KEY, name varchar(10) ) 数据10 ...

  5. 转:一个经典例子让你彻彻底底理解java回调机制

    一个经典例子让你彻彻底底理解java回调机制 转帖请注明本文出自xiaanming的博客(http://blog.csdn.net/xiaanming/article/details/17483273 ...

  6. 一个经典例子让你彻彻底底理解java回调机制

    转帖请注明本文出自xiaanming的博客(http://blog.csdn.net/xiaanming/article/details/17483273),请尊重他人的辛勤劳动成果,谢谢 所谓回调: ...

  7. 回调--一个经典例子让你彻彻底底理解java回调机制

    本文出自xiaanming的博客(http://blog.csdn.net/xiaanming/article/details/17483273),请尊重他人的辛勤劳动成果,谢谢 以前不理解什么叫回调 ...

  8. 【C语言学习笔记】空间换时间,查表法的经典例子!知识就是这么学到的~

    我们怎么衡量一个函数/代码块/算法的优劣呢?这需要从多个角度看待.本篇笔记我们先不考虑代码可读性.规范性.可移植性那些角度. 在我们嵌入式中,我们需要根据实际资源的情况来设计我们的代码.比如当我们能用 ...

  9. Conquer and Divide经典例子之汉诺塔问题

    递归是许多经典算法的backbone, 是一种常用的高效的编程策略.简单的几行代码就能把一团遭的问题迎刃而解.这篇博客主要通过解决汉诺塔问题来理解递归的精髓. 汉诺塔问题简介: 在印度,有这么一个古老 ...

随机推荐

  1. notepad++安装nppFTP

    官网下载的最新版notepad++,结果pluginadmin里面installnppftp总是安不上,点击install之后然后点是就退出,再进去也没有安装好. 网上找了半天也没找到什么有用信息,最 ...

  2. 如何给PDF文件制作书签

    书本阅读的时候我们有时候会制作一些漂亮的书签,那么电子文档也是有书签的,要怎么制作小伙伴们都知道吗?应该会有许多的小伙伴还不知道,今天就为大家分享一下电子文件如何添加书签的.就以PDF这个现在常用的电 ...

  3. <转>Go语言TCP Socket编程

    授权转载: Tony Bai 原文连接: https://tonybai.com/2015/11/17/tcp-programming-in-golang/ Golang的主要 设计目标之一就是面向大 ...

  4. OpenCV-Python:轮廓

    啥叫轮廓 轮廓是一系列相连的点组成的曲线,代表了物体的基本外形. 轮廓与边缘很相似,但轮廓是连续的,边缘并不全都连续,其实边缘主要是作为图像的特征使用,比如用边缘特征可以区分脸和手,而轮廓主要用来分析 ...

  5. 基于 EntityFramework、Autofac 的 UnitOfWork 框架(一)

    之前公司项目参考 NopCommerce 开发了一套系统,但是不支持 UnitOfWork,最近想开发新的项目,所以就基于原有的基础上又添加 UnitOfWork 支持,由于目前正在逐步完善中,所以可 ...

  6. python 项目实例

    参考:  https://blog.csdn.net/yz764127031/article/details/71522161 https://www.cnblogs.com/linuxprobe/p ...

  7. Linux进程和端口互相查看方法

    一.查找应用进程的端口 ps -ef |grep mysql 查看进程 ps:将某个进程显示出来 -A 显示所有程序.  -e 此参数的效果和指定"A"参数相同. -f 显示UID ...

  8. iOS异常捕获和处理

    2013年4月份整理的代码,仅作记录:   //先宏定义 //发布和未发布状态的日志切换 #ifdef DEBUG     //异常栈开关     #define STACK_KEY YES     ...

  9. 3dmax 笔记本电脑

    一般来说,苹果MacBook系列适合运行3ds Max. 3ds Max是一款大型图像处理软件,对电脑硬件要求高.在所有MacBook系列中,配备Retina屏幕的MacBook Pro 15寸是最佳 ...

  10. 教你如何用勾勾街快速生成一个苹果IOS APP

    现在苹果手机上的各种各样的APP,想不想也有一款属于自己的专属APP?很简单,用勾勾街可以在3分钟内快速制作一款,快来看看! 工具平台: 勾勾街 (www.gogojie.com ) 操作方法: 1. ...