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;
}





猜你喜欢

转载自blog.csdn.net/qq_23335927/article/details/83151760