指针参数传递和调用栈

可以通过跟踪来了解指针在参数传递时,内存空间的栈调用状态。以下的内存空间地址值均为在我机器上调试时的值,不同的机器可能值会不一样。

代码1:
C++代码

void GetMemory(char *p)  
{  
    p = (char*)malloc(100);  
}  
int main(int argc, char *argv[])  
{  
    char *str = NULL;  
    GetMemory(str);  
    strcpy(str, "Hello");  
    return 0;  
}  

str没有得到分配内存的地址值。
内存空间状态:首先申请了四个字节的栈空间,存放str指针,此时str的值为0,存放str的这块内存的地址值为0x0012ff7c。调用函数 GetMemory,指针P入栈,也分配了四个字节的栈空间,P被赋str的值即此时P的值也为0,存放指针P的内存地址是0x0012ff2c。然后将新开辟的100个字节的内存空间地址赋给P,此时P的值为0x00372b70。函数调用结束时str的值仍为0,str并没有得到那块100个字节的内存空间地址值!

代码2:
C++代码

void GetMemory(char **p)  
{  
    *p = (char*)malloc(100);  
}  
int main(int argc, char *argv[])  
{  
    char *str = NULL;  
    GetMemory(&str);  
    strcpy(str, "Hello");  
    return 0;  
}  

str可以得到分配内存的地址值。
内存空间状态:首先申请了四个字节的栈空间,存放str指针,此时str的值为0,存放str的这块内存的地址值为0x0012ff7c。调用函数 GetMemory,指针P入栈,也分配了四个字节的栈空间,此时P是一个二级指针,存放了指针str的地址值,即P的值是0x0012ff7c,存放指针P的内存空间的地址值是0x0012ff2c。然后将新开辟的100个字节的内存空间地址值0x00372b70赋给*P,即str,所以str的值为 0x00372b70。函数返回时str的值为分配的100个字节的内存空间的地址!

代码3:
C++代码

void GetMemory(char **p)  
{  
    // 这条语句编译出错,将一个二级指针指向分配的地址了  
    // p = (char*)malloc(100);  
    // 可以使用强制转换,但程序crash  
    p = reinterpret_cast(malloc(100));  
}  
int main(int argc, char *argv[])  
{  
    char *str = NULL;  
    GetMemory(&str);  
    strcpy(str, "Hello");  
    return 0;  
}  

str不能得到分配内存的地址值,程序crash。
如果在GetMemory函数中使用如下语句 p = (char)malloc(100); 会出现编译出错,原因是不能将一个二级指针指向分配的内存空间地址。如果使用强制转换 p = reinterpret_cast*>(malloc(100)); 编译可以通过,但是会造成程序崩溃。

代码4:
C++代码

void GetMemory(char *p)  
{  
    p = (char*)malloc(100);  
}  
int main(int argc, char *argv[])  
{  
    char *str = NULL;  
    GetMemory(&str); // 这条语句会编译出错,将一个指针地址值传给了一级指针  
    strcpy(str, "Hello");  
    return 0;  
}  

str不能得到分配内存的地址值,编译出错。 将一个指针的地址值传给了一级指针,非法操作!

你可能感兴趣的:(指针参数传递和调用栈)