C++函数形参传指针的再次讨论

2014-11-24 08:35:56 · 作者: · 浏览: 0

先来看一段代码:

void GetMemory(char *p)
{
	p = new char[20];
}

void Test()
{
	char * str = NULL;
	GetMemory(str);
	strcpy(str,"hello,world");
	
	delete str;
	str = NULL;
}

每一个程序员在面试中都会遇到的问题,上面那段代码问题在哪里呢?

(1) 会造成内存泄漏

(2) 在strcpy会出现程序奔溃。

以下是具体的问题分析:

我们知道函数一般都是传值,即使是指针的值,也是一个数值而已。下面通过调试来看看str的值的变化:


\

第1步刚刚进入Test函数,str分配好地址0x0012f464,但是里面的值为0xcccccccc,即为没有初始化的值;
< http://www.2cto.com/kf/ware/vc/" target="_blank" class="keylink">vcD4KPHA+PGJyPgo8L3A+CjxwPjxicj4KPC9wPgo8cD48aW1nIHNyYz0="https://www.cppentry.com/upload_files/article/49/1_orffz__.jpg" alt="\">

第2步在Test函数中,str被赋值为NULL即,此时str的地址0x0012f464,但是里面的值为0x00000000,注意因为此时没有进入子函数GetMemory,所以p还没有分配内存;



\

第3步进入子函数GetMemory中,str的值被传递给新申请的p。此时p的地址0x0012f384,但是里面的值为0x00000000;



\

第4步在子函数GetMemory中,在堆上申请的 new char[20]返回的地址值保存在p中。此时p的值已经由之前的0x0012f384变为0x0035cf08;




\

第5步在跳出子函数GetMemory,返回到Test函数中,注意观察此时的变量状态:临时变量p的生命周期随着子函数GetMemory的结束一起结束了,并没有把新申请的new char[20]的地址传递到Test函数中来;str的值还是之前的,没有做任何改变。

通过上面的一系列调试截图,我们知道函数传递时仅仅是传递的值。借用一位网友的总结:

“函数的参数进行值拷贝,即使传的是指针,也的对指针(即指针里存的地址)的拷贝。”

遇到搞不清楚的时候勤动手,多调试,肯定会搞明白的。

可能聪明的你还想追问,那指针做函数参数是怎么用的呢?

其实我们之所以使用指针,仅仅是因为通过它作为纽带可以找到我们想操作的东西,大多数时候都是通过*p="N';或者p[0]='N';或者*(p++)='N';之类用法来修改指针指向的地址里面的变量值,而不是修改指针的值。更多的知识请读者自己在使用中慢慢去体会了。


参考网址列表:

http://www.cppblog.com/mzty/archive/2006/07/07/9531.html

http://www.cnblogs.com/nerohwang/p/3481725.html