C语言面试题---指针篇(三)

版本声明:本文转载于公众号TeachPlus
C语言面试题---指针篇(三)
了解了内存空间,接下来我们就一起看一下指针自身用法的一些题目,先来看这样一道题目:
分析下面代码:`
# include <stdio.h>
# include <stdlib.h>
# include <string.h>
 
void  getmemory( char *p)
{
    p=( char *) malloc(100);
    strcpy(p,"hello world");
}
 
int  main( void)
{
    char *str=NULL;
    getmemory(str);
    printf("%s\n",str);
    free(str);
    return 0;
}

本题解析

答案:程序崩溃, getmemory  中的 malloc  不不能返回动态内存,  free()对 str 操作很危险,程
序会崩溃,出现段错误
这道题目初一看跟我们前面的一道题目非常相像,但是却又截然不同。
在该程序中, getmemory 中 p 是形参,所谓形参在运行中会产生一个临时变量,只会把外界传入
的参数的值接收到,所有的改变不会影响外界的实际参数。 getmemory  函数中,因为我们要改变
传入的str指针的指向,也就是说要改变str指针变量的值,应该传入的是指针变量的地址。
因此在函数的形参中不应该写 char *p  ,而是应该是 char **p  。
getmemory(str)  调用后,传入的是指针变量保存的对象地址,
 p=(char *) malloc(100)  实际上是把申请的动态内存空间的首地址付给p这个临时变量,
改变了p的指向,对于外界的str是没有影响的,因此这个是错误的。
应该修改成指向指针的指针void getmemory(char *p) ,这样 malloc 返回的地址付给 p(即str变量本身)。

相关知识点

来看另外的一道题目:
下面函数有什么问题,应该怎么修改
char *strA()
{
    char  str[] ="hello world";
    return  str;
}

分析:

因为这个函数返回的是局部变量的地址,当调用这个函数后,这个局部变量str就释放了,
所以返回的结果是不不确定的且不不安全,随时都有被收回的可能。
这个str里存在的地址是函数strA栈里“hello world”的首地址。
函数调用完成,栈帧恢复调用strA之前的状态,临时空间被重置,堆栈“回缩”,
strA栈帧不再属于应该访问的范围。
这段程序可以正确输出结果,但是这种访问方法违背了函数的栈帧机制。
但是只要另外一个函数调用的话,你就会发现,这种方式的不合理及危险性。

如果想获得正确的函数,改成下面这样就可以:
char *strA()
{
    char * str ="hello world";
    return  str;
}
首先要搞清楚char *str 和 char str[] :
1、
char  str[] = "hello world";
  
  
是分配一个局部数组。局部数组是局部变量,它所对应的是内存中的栈。局部变量的生命周期结
束后该变量不存在了。
2、
char * str = "hello world";
  
  

是指向了了常量区的字符串,位于静态存储区,它在程序生命期内恒定不变,所以字符串还在。
无论什么时候调用 strA,它返回的始终是同一个“只读”的内存块。
char *strA()
{
    static  char  str[] ="hello world";
    return  str;
}




  • 0
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值