首先要明白函数体是一个作用域,函数的形参为函数提供了已命名的局部存储空间。形参是在函数的形参表中定义的,由调用函数时传递给函数的实参初始化。每次调用函数时,都会重新创建该函数所有的形参,此时所传递的实参将会初始化对应的参数。形参的初始化与变量的初始化一样:如果形参具有非引用类型,则复制实参的值;如果形参为引用类型,则它是实参的别名。
void GetMemory(char *p)
{
p=(char*)malloc(100);
}
void Test(void)
{
char *str = NULL;
GetMemory(str);
strcpy(str,”helloworld”);
printf(str);
}
请问运行Test函数会有什么样的结果?
答:程序崩溃。因为在调用GetMemory的时候,用实参str的拷贝来初始化形参p,函数体中只是改变了p(str的拷贝)的值,并没有改变str的值。实际上str此时仍为一个无效的指针。
char *GetMemory(void)
{
char p[]=”helloworld”;
return p;
}
void Test(void)
{
char *str = NULL;
str = GetMemory();
printf(str);
}
请问运行Test函数会有什么样的结果?
答:可能是乱码。因为GetMemory返回的是指向“栈内存”的指针,该指针的地址不是NULL,但其原先的内容已经被清除,新内容不可知。
当函数执行完时,就会释放分配给局部对象的存储空间(栈内存)。此时,对局部对象的引用和指针就会指向不确定的内存。
void GetMemory(char **p, int num)
{
*p = (char*)malloc(num);
}
void Test(void)
{
char *str = NULL;
GetMemory(&str, 100);
strcpy(str, “hello”);
printf(str);
}
请问运行Test函数会有什么样的结果?
答:(1)能够输出hello,因为在调用GetMemory的时候,用实参str的拷贝来初始化形参p,函数体改变了str拷贝所指向的内存的值(2)内存泄漏