子函数不能传递动态内存的原因解析

来源:互联网 发布:windows bug 编辑:程序博客网 时间:2024/05/04 09:12

举例:

voidGetMemory(char *p)
{
     p = (char *)malloc(100);
}
void Test(void)
{
    char *str = NULL;
    GetMemory(str);
    strcpy(str, "hello world");
    printf(str);
}

请问运行Test 函数会有什么样的结果?
答:程序崩溃(段错误)。因为GetMemory 并不能传递动态内存,Test 函数中的str 一直都是 NULL。strcpy(str, "helloworld");将使程序崩溃。



我们知道,通过函数传递动态内存的方式有两种:返回内存的地址或者使用二级指针。用返回值的方式很容易理解,可为什么不能使用一级指针呢?

      《高质量c/c++程序设计》里面给出了理由:编译器会为每个参数制作临时副本,指针参数p的副本是_p并使_p = p。如果函数体内的程序修改了_p 的内容,就导致参数p 的内容作相应的修改。这就是指针可以用作输出参数的原因。如果,_p 申请了新的内存,只是把_p 所指的内存地址改变了,但是p 丝毫未变,所以不能传递动态内存。
      深刻理解上面的话很重要,为了说明这个问题,我们举以下的例子:


#include <stdio.h>   
#include <stdlib.h>   
   
void func(int *p)   
{   
    printf("in func:\n");   
    printf("before malloc:\n");   
    printf("&p = %p, p = %p\n", &p, p);   
       
    p = (int*)malloc(1024);   
       
    printf("after malloc:\n");   
    printf("&p = %p, p = %p\n", &p, p);   
}   
   
   
int main()   
{   
    int *p = (int *)malloc(3);   
       
    printf("in main:\n");   
    printf("&p = %p, p = %p\n", &p, p);   
    func(p);   
   
    printf("in main:\n");      
    printf("&p = %p, p = %p", &p, p);   
    free(p); 
    return 0;   
}  
 



0 0
原创粉丝点击