先来看一段代码:
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