C语言函数中变量的内存问题
看见两道题目:有点疑问第一题:char*GetMemory(void){charp[]="helloworld";returnp;}voidTest(void){char...
看见两道题目:有点疑问
第一题:
char *GetMemory(void)
{
char p[] = "hello world";
return p;
}
void Test(void)
{
char *str = NULL;
str = GetMemory();
printf(str);
}
请问运行Test函数会有什么样的结果?
答:可能是乱码。 char p[] = "hello world"; return p; 的p[]数组为函数内的局部自动变量,在函数返回后,内存已经被释放。这是许多程序员常犯的错误,其根源在于不理解变量的生存期。
第二题:
编写strcpy函数(10分)
已知strcpy函数的原型是
char *strcpy(char *strDest, const char *strSrc);
其中strDest是目的字符串,strSrc是源字符串。
(1)不调用C++/C的字符串库函数,请编写函数strcpy
char *strcpy(char *strDest, const char *strSrc);
{ assert((strDest!=NULL) && (strSrc !=NULL)); // 2分
char *address = strDest; // 2分while( (*strDest++ = * strSrc++) != ‘\0’ ) // 2分
NULL ;
return address ; // 2分
}
(2)strcpy能把strSrc的内容复制到strDest,为什么还要char * 类型的返回值?
答:为了实现链式表达式。 // 2分
例如 int length = strlen( strcpy( strDest, “hello world”) );
我的疑问是:
第一题的解答中说函数中的变量是有生存周期的,函数返回后内存已经被释放,那么第二题答案中 为了实现链式表达式,所以返回address,return address 。我的疑问是按照第一题的说法,address在函数返回时已被释放了啊 ,strlen(address)就不对了啊
菜鸟 求指教
非常 特别 感谢大家的回答 展开
第一题:
char *GetMemory(void)
{
char p[] = "hello world";
return p;
}
void Test(void)
{
char *str = NULL;
str = GetMemory();
printf(str);
}
请问运行Test函数会有什么样的结果?
答:可能是乱码。 char p[] = "hello world"; return p; 的p[]数组为函数内的局部自动变量,在函数返回后,内存已经被释放。这是许多程序员常犯的错误,其根源在于不理解变量的生存期。
第二题:
编写strcpy函数(10分)
已知strcpy函数的原型是
char *strcpy(char *strDest, const char *strSrc);
其中strDest是目的字符串,strSrc是源字符串。
(1)不调用C++/C的字符串库函数,请编写函数strcpy
char *strcpy(char *strDest, const char *strSrc);
{ assert((strDest!=NULL) && (strSrc !=NULL)); // 2分
char *address = strDest; // 2分while( (*strDest++ = * strSrc++) != ‘\0’ ) // 2分
NULL ;
return address ; // 2分
}
(2)strcpy能把strSrc的内容复制到strDest,为什么还要char * 类型的返回值?
答:为了实现链式表达式。 // 2分
例如 int length = strlen( strcpy( strDest, “hello world”) );
我的疑问是:
第一题的解答中说函数中的变量是有生存周期的,函数返回后内存已经被释放,那么第二题答案中 为了实现链式表达式,所以返回address,return address 。我的疑问是按照第一题的说法,address在函数返回时已被释放了啊 ,strlen(address)就不对了啊
菜鸟 求指教
非常 特别 感谢大家的回答 展开
3个回答
展开全部
你问这个问题很好,这么给你说吧,第一题和第二题都是返回的指针,都能正确的得到返回的指针值,但是这两指针所指的地址对应的内存单元中的值却不一样了,对于第一题,指针对应的存储单元是通过函数之外定义的,然后传入这个函数做处理,那么对于函数之外传入的参数对于这个函数而言肯定是全局,那么在此函数执行完毕,释放时,不再影响这个全局的参数了,但是第二题,这指针对应的单元是在函数内部申请的,那么,当函数执行完毕,就释放在那内部所释放的内存,就是说,这块内存被系统认为是没被占用的,那么系统可能在执行其他任务时就有把这部分内存使用了,但是你返回的指针不变,还是指向这块区域,但是这块区域的内容已经不是以前的了,可能被其他任务改成其他的值了! 建议你去查一下桟内存和动态分配内存的区别,!
展开全部
你看一下汇编代码就明白了,目前大部分编译器都把指针的大小定为32位,也就是CPU中通用寄存器的长度,
return address编译后的汇编代码基本上是:
mov eax,address
leave
ret
其中leave语句是起堆栈平衡作用的,不用管。
这样函数结束后,虽然各个内部变量都已经被释放了,但是address的值还存在于eax寄存器中,在主调函数中可以立即取得eax的值,也就是address的值
return address编译后的汇编代码基本上是:
mov eax,address
leave
ret
其中leave语句是起堆栈平衡作用的,不用管。
这样函数结束后,虽然各个内部变量都已经被释放了,但是address的值还存在于eax寄存器中,在主调函数中可以立即取得eax的值,也就是address的值
已赞过
已踩过<
评论
收起
你对这个回答的评价是?
展开全部
地一个是局部变量,返回的是函数内部参数的地址;
第二个是函数参数
返回的是实参的地址。
第二个是函数参数
返回的是实参的地址。
已赞过
已踩过<
评论
收起
你对这个回答的评价是?
推荐律师服务:
若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询