手机站
网通分站
电信主站
密 码:
用户名:
当前位置 : 主页>程序设计>C/C++>列表

论C/C 函数间动态内存的传递

来源:互联网 作者:west263.com 时间:2008-02-23
西部数码-全国虚拟主机10强!40余项虚拟主机管理功能,全国领先!双线多线虚拟主机南北访问畅通无阻!免费赠送企业邮局,.CN域名,自助建站480元起,免费试用7天,满意再付款! P4主机租用799元/月.月付免压金!
当您涉及到C/C 的核心编程的时候,您会无止境地和内存管理打交道.这些往往会使人受尽折磨.所以假如您想深入C/C 编程,您必须静下心来,好好苦一番.
现在我们将讨论C/C 里我认为哪一本书都没有完全说清楚,也是涉及概念细节最多,语言中最难的技术之一的动态内存的传递.并且在软件研发中很多专业人员并不能写出相关的合格的代码.

【引入】 看下面的例子,这是我们在编写库函数或项目内的一起函数经常希望的.
void MyFunc(char *pReturn, size_t size)
{
………
pReturn = (char *)malloc(sizeof(char) * num);
………
}
我们能够很明显地看出代码作者的意图,他想在函数调用处声明一个指针
char *pMyReturn=NULL;
然后调用MyFunc处理并返回一段长度为size的一段动态内存.
那么作者能达到预期的效果吗?
那么我能够告诉作者,他的程式在编译期很幸运地通过了,可是在运行期他的程式崩溃终止.
原因何在,是他触犯了系统不可侵犯的条款:错误地操作内存.


【内存操作及问题相关知识点】为了能完全解决动态内存传递的问题,我们先回顾一下内存管理的知识要点.
(1)内存分配方式有三种:
●从静态存储区域分配。内存在程式编译的时候就已分配好,这块内存在程式的整个运行期间都存在。例如全局变量,static变量。
●在栈上创建。在执行函数时,函数内局部变量的存储单元都能够在栈上创建,函数执行结束时这些存储单元自动被释放。栈内存分配运算内置于处理器的指令集中,效率很高,但是分配的内存容量有限。
●从堆上分配,亦称动态内存分配。程式在运行的时候用malloc或new申请任意多少的内存,程式员自己负责在何时用free或delete释放内存。动态内存的生存期由我们决定,使用很灵活。
(2)指针的操作流程
申请并初始化或配置为空:int *pInt=NULL;
开辟空间或使其指向对象:pInt=new Int(3);或int i=3;pint=&i;
用指针(更确切地说是操作内存,在使用之前加if(pint!=NULL)或assert(pInt!=NULL)后再使用,以防内存申请失败的情况下使用指针):
if(p!=NULL) {use pint};
释放使用完的内存.free(pInt);
置指针为空pInt=NULL;(避免野指针的出现)
(3) 在函数的参数传递中,编译器总是要为函数的每个参数制作临时副本,假如参数为p的话,那么编译器会产生p的副本_p,使_p=p; 假如函数体内的程式修改了_p的内容,就导致参数p的内容作相应的修改。这就是指针能够用作输出参数的原因.


【问题分析】
根据上面的规则我们能够很容易分析例子中失败的原因.

void MyFunc(char *pReturn, size_t size)
{
………
pReturn = (char *)malloc(sizeof(char) * num);
………
}
void main(void){
char *pMyReturn=NULL;
MyFunc(pMyReturn,10);
}
在MyFunc(char *pReturn, size_t size)中_pMyReturn真实地申请到了内存, _pMyReturn申请了新的内存,只是把_pMyReturn 所指的内存地址改变了,但是pMyReturn丝毫未变。所以函数MyFunc并不能输出任何东西。事实上,每执行一次MyFunc就会泄露一块内存,因为没有用free释放内存。

【问题解决方案】
函数间传递动态数据我们能够有三种解决方法.
方法一.假如我们是用C 编程,我们能够很方便地利用引用这个技术.我也极力推荐您用引用,因为他会使您少犯一些错误.以下是个例子.
void MyFunc(char* &pReturn,size_t size){
pReturn=(char*)malloc(size);
memset(pReturn,0x00,size);
if(size>=13)
strcpy(pReturn,"Hello World!");
}

void main(){
char *pMyReturn=NULL;
MyFunc(pMyReturn,15);
if(pMy Return!=NULL)
{
char *pTemp=pMyReturn;
while(*pTemp!=''\0'')
cout<<*pTemp ;
pTemp=NULL;
strcpy(pMyReturn,"AAAAAAAA");
free(pMyReturn);
pMyReturn=NULL;
}
}
方法二.利用二级指针
void MyFunc (char ** pReturn, size_t size)
{
* pReturn = (char *)malloc(size);
}
void main(void)
{
char * pMyReturn = NULL;
MyFunc (&pMyReturn, 100);// 注意参数是 & pMyReturn
if(pMyReturn!=NULL){
strcpy(pMyReturn, "hello");
cout<< pMyReturn << endl;
free(pMyReturn);
pMyReturn=NULL;
}}
为什么二级指针就能够了.原因通过函数传递规则能够很容易地分析出来.我们将& pMyReturn传递了进去,就是将双重指针的内容传递到了函数中.函数过程利用改变指针的内容,这样pMyReturn很明显指向了开辟的内存 .

方法三. 用函数返回值来传递动态内存

char * MyFunc (void)
{
char *p =new char[20];
memset(p,0x00,sizeof(p));
return p;
}
void main(void)
{
char *str = NULL;
str = MyFunc();
if(str!=NULL)
{
strcpy(str,"Hello,baby");
cout<< str << endl;
free(str);
str=NULL;
}

}
请注意的是函数写成这样的话
char * MyFunc (void)
{
char *p =”Hello World”
return p;
}
的话,您是不能返回什么动态内存的,因为p指向的是字符串常量.内存在位于静态存储区
上分配,您无法改变.(您想要得到动态内存我们一定要看到malloc或new).




文章整理:西部数码--专业提供域名注册虚拟主机服务
http://www.west263.com
以上信息与文章正文是不可分割的一部分,如果您要转载本文章,请保留以上信息,谢谢!

热点关注
IDC资讯 虚拟主机 域名注册 托管租用 vps主机 智能建站
网站运营 建站经验 策划盈利 搜索优化 网站推广 免费资源
网站联盟 联盟新闻 联盟介绍 联盟点评 网赚技巧
行业资讯 业界动态 搜索引擎 网络游戏 门户动态 电子商务 广告传媒
网络编程 Asp.Net编程 Asp编程 Php编程 Xml编程 Access Mssql Mysql 其它
服务器技术 Web服务器 Ftp服务器 Mail服务器 Dns服务器 安全防护
软件技巧 其它软件 Word Excel Powerpoint Ghost Vista QQ空间 QQ FlashGet 迅雷 Internet Explorer
网页制作 FrontPages Dreamweaver Javascript css photoshop fireworks Flash
程序设计 Java技术 C/C++ VB delphi
网络知识 网络协议 网络安全 网络管理 组网方案 Cisco技术
操作系统 Win2000 WinXP Win2003 Mac OS Linux FreeBSD
返回首页 |关于我们 | 联系我们 | 付款方式 | 创业联盟 | 价格总览 | 资讯中心 | 友情链接 | 网站地图 | 招贤纳士 | RSS