malloc和strcpy,入门的指针面试题

2023-11-18

指针是C和C++编程语言中一个重要的概念,因此在面试以及工作中经常会涉及到指针相关的问题,现在列举几个比较基础问题。

一、

void getMemory( char *p )
{
 p = (char *) malloc( 100 );
}
void test( void ) 
{
 char *str = NULL;
 getMemory( str ); ///
 strcpy( str, "hello world" );
 printf( str );
}

这段代码有一个问题是在函数getMemory中,对于初学者来说,肯定想当然认为,类型一样,还有什么问题?指针p被重新赋值为一个新的内存空间的地址,但是这个修改不会影响到函数外部的实参str指向的内存空间,相当于传了NULL,因为是要改变这个值的指向,要传他的本身的指针,也就是指针的指针。因此,在调用strcpy函数时,传递的是一个未初始化的指针,会导致未定义的行为。

修复这个问题的方式有两种:

将函数GetMemory返回分配的内存指针,并将其赋值给str指针。

char * getMemory( void )
{
    char * p = ( char * )malloc( 100 );
    return p;
}

void test( void ) 
{
 char * str = NULL;
 str = getMemory(); 
 strcpy( str, "hello world" );
 printf( str );
}

int main()
{
  test();
  return 0;
}

将函数GetMemory参数改为指向指针的指针类型,并通过间接寻址方式来修改指针的值。

void getMemory( char ** p )
{
    *p = ( char * )malloc( 100 );
}

void test( void ) 
{
 char *str = NULL;
 getMemory( &str ); //遇到这种二级指针,在数据结构二叉树也是经常遇到的
 strcpy( str, "hello world" );
 printf( str );
}

int main()
{
  test();
  return 0;
}

二叉树的每个节点都有两个子节点,因此二叉树可以用指向节点的指针表示。在函数中需要修改指向节点的指针时,我们可以使用指向指针的指针(即二级指针)来进行操作。以下是构建二叉树的示例代码:

#include <stdio.h>
#include <stdlib.h>

// 定义二叉树节点结构体
typedef struct TreeNode {
    int val;
    struct TreeNode *left;
    struct TreeNode *right;
} TreeNode;

// 向二叉树中插入一个新节点
void insert(TreeNode **root, int val) {
    if (*root == NULL) {
        // 当前节点为空,创建一个新节点
        *root = (TreeNode *)malloc(sizeof(TreeNode));
        (*root)->val = val;
        (*root)->left = NULL;
        (*root)->right = NULL;
    } else if (val < (*root)->val) {
        // 新节点值小于当前节点值,递归插入左子树
        insert(&((*root)->left), val);
    } else {
        // 新节点值大于等于当前节点值,递归插入右子树
        insert(&((*root)->right), val);
    }
}

// 中序遍历二叉树
void inorder(TreeNode *root) {
    if (root == NULL) {
        return;
    }
    inorder(root->left);
    printf("%d ", root->val);
    inorder(root->right);
}

int main() {
    TreeNode *root = NULL; // 创建一棵空树
    insert(&root, 5);
    insert(&root, 3);
    insert(&root, 7);
    insert(&root, 1);
    insert(&root, 9);
    inorder(root); // 中序遍历输出:1 3 5 7 9
    return 0;
}

在这个示例代码中,我们定义了一个二叉树节点结构体,并使用指向指针的指针(即二级指针)来插入新节点和修改树的结构。

二、

void test( void )
{
 char *str = (char *) malloc( 100 );
 strcpy( str, "hello" );
 free( str ); 
 ... //省略的其它语句
}

这段代码中存在一个问题,即在调用完free函数之后,指针str仍然指向了被释放的内存空间。虽然这个指针已经被释放,但是它的值并没有被修改,因此在继续使用该指针时可能会导致未定义的行为,例如访问非法内存、数据损坏等。为避免这个问题,我们可以在调用free函数之后将指针str赋值为NULL,这样在继续使用该指针时就会导致段错误(segmentation fault)等异常,这个经常在工作中会遇到,从而更容易发现问题。

三、

void test()
{
int *p = (int *)malloc(INT_MAX/4);
*p = 20;//如果p的值是NULL,就会有问题   进行判断是否为空
free(p);
}

这段代码中存在一个问题,即在申请内存空间时使用了INT_MAX/4这么大的内存,这可能导致失败或者内存不足,从而导致malloc函数返回NULL。此时如果直接对指针进行解引用赋值操作,会导致程序运行异常。

为避免这个问题,我们需要在调用malloc函数之后判断其返回值是否为NULL,以确保分配内存成功。修复这个问题的方式:

void test()
{
  int *p = (int *)malloc(INT_MAX/4);
  if (p == NULL) {
    printf("Failed to allocate memory\n");
    return;
  }
  *p = 20;
  free(p);
}

四、

void test()
{
  int i = 0;
  int *p = (int *)malloc(10*sizeof(int)); //分配40个字节空间
  if(NULL == p)
  {
    exit(0); //分配失败就退出
  }
  for(i=0; i<=10; i++) 
  {
    *(p+i) = i;//当i是10的时候越界访问
  }
    free(p);
  }

这段代码存在一个问题,即在使用for循环赋值时,数组下标从0到10共11个,而p指向的内存空间只有10个int型变量(40字节),也就是说当i等于10时数组已经越界访问,这会导致未定义的行为。

修复这个问题的方式是将for循环中的判断条件修改为i<10,即数组下标最大只能是9,不会越界访问。例如:

void test()
{
  int i = 0;
  int *p = (int *)malloc(10*sizeof(int)); //分配40个字节空间
  if(NULL == p)
  {
    exit(0); //分配失败就退出
  }
  for(i=0; i<10; i++) 
  {
    *(p+i) = i;
  }
    free(p);
}

五、

void test()
{
int *p = (int *)malloc(100);
p++;
free(p);//p不再指向动态内存的起始位置
} 

这段代码存在一个问题,即在调用free函数时,指针p已经不再指向动态内存的起始位置,这会导致free函数释放的内存空间不正确,如果在释放内存之前修改了指向该内存的指针,则可能导致内存泄漏或者程序崩溃,不能正确进行释放空间,因为其实地址不见,free就找不到正确释放内存的大小。在调用malloc()或calloc()函数时,系统会自动记录该内存块的大小,以便在调用free()函数时准确地释放相应的内存。

六、

char *getMemory(void)
{
  char p[] = "hello world";
  return p;
}
void test(void)
{
  char *str = NULL;
  str = getMemory();
  printf(str);
}
int main()
{
  test();
  return 0;
}

这段代码有一个问题是返回了一个局部变量的地址,具体来说,函数getMemory中定义了一个字符数组p, 并将其初始化为字符串"hello world",然后将p的地址作为返回值。但是由于p是一个局部变量, 在函数返回时它所占用的内存会被回收,因此在调用getMemory函数并将返回值赋给指针str后, str指向的内存空间已经无效,访问该空间就会导致未定义的行为。修复这个问题的方式是将p声明为静态数组或者使用动态内存分配函数(如malloc)分配堆上的内存空间。例如:

char *getMemory(void)
{
  static char p[] = "hello world";
  return p;
}
void test(void)
{
  char *str = NULL;
  str = getMemory();
  printf("%s", str);
}
int main()
{
  test();
  return 0;
}

七、

以下是一个错误使用 strcpy 函数的代码题:

#include <stdio.h>
#include <string.h>

int main(void) {
    char src[10] = "Hello!";
    char dest[5];

    // 错误示范:dest 的大小只有 5,无法容纳源字符串 "Hello!"
    strcpy(dest, src);

    printf("src: %s\n", src);
    printf("dest: %s\n", dest);

    return 0;
}

在这个例子中,我们声明了两个字符数组 src 和 dest,并将字符串 “Hello!” 存储在 src 中。然后使用 strcpy 函数将 src 中的字符串复制到 dest 中。由于 dest 的大小只有 5,无法容纳源字符串 “Hello!” 的长度,所以会导致缓冲区溢出问题,可能导致程序崩溃或产生不可预知的行为。

正确的做法是要确保目标字符串具有足够的空间来存储源字符串,可以使用 strncpy 函数代替 strcpy 函数,并指定要复制的最大字符数,以避免缓冲区溢出问题。例如:

#include <stdio.h>
#include <string.h>

int main(void) {
    char src[10] = "Hello!";
    char dest[5];

    // 使用 strncpy 函数将 src 中的字符串复制到 dest 中
    strncpy(dest, src, sizeof(dest)-1);
    dest[sizeof(dest)-1] = '\0';

    printf("src: %s\n", src);
    printf("dest: %s\n", dest);

    return 0;
}

使用 strncpy 函数时,我们指定要复制的最大字符数为 sizeof(dest)-1,这样可以保证在复制字符串时不会超出 dest 的大小。另外,由于 strncpy 函数不会自动在目标字符串末尾添加 ‘\0’ 字符,所以需要手动将其添加到 dest 的末尾. 以上是面试中经常被问。

如果您喜欢今天的文章,请不要忘记在下方留言并分享给您的朋友们。我将非常感激您的支持,并期待您的关注,共同探索更多有趣的话题,我是 小昭debug。

希望以上小结能够对您有所帮助!

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

malloc和strcpy,入门的指针面试题 的相关文章

随机推荐

  • STP与RSTP区别

    STP 不能快速迁移 即使是在点对点链路或边缘端口 边缘端口指的是该端口直接与用户终端相连 而没有连接到其它设备或共享网段上 也必须等待2 倍的ForwardDelay 的时间延迟 端口才能迁移到转发状态 RSTP Rapid Spanni
  • Python 异步: 异步上下文管理器(17)

    动动发财的小手 点个赞吧 上下文管理器是一种 Python 构造 它提供了一个类似 try finally 的环境 具有一致的接口和方便的语法 例如通过 with 表达 它通常与资源一起使用 确保在我们完成资源后始终关闭或释放资源 无论资源
  • VC++ OpenCV+ZBar二维码识别

    利用OpenCV处理图像的优势 结合ZBar提高二维码识别结果 接口定义 include
  • SpringSecurity配置类--常用配置

    SpringSecurity配置类 在学习这门课的时候 实现各种功能时进行了各种配置 我想将各种配置综合讲述一下 首先自定义配置类 需要继承WebSecurityConfigurerAdapter这个类 在这个类里面做了一些默认配置 Con
  • linux grpc,grpc linux下的编译使用-Go语言中文社区

    1 一些工具安装 apt get install build essential autoconf libtool pkg config apt get install libgflags dev libgtest dev apt get
  • [转]IDEA 撤销 merge 操作(详解)

    目录 一 前言 二 解决方案 通过 Git Bash 命令行解决 1 合并过程中未发生冲突 2 合并过程中发生冲突 三 解决方案 通过 IDEA 解决 附带详细的操作图 1 合并过程中未发生冲突 2 合并过程中发生冲突 四 最后 作为一个开
  • 自己制作的 macOS Mojave 10.15.3 iso 文件,亲测可用(附 VMware15 安装 macOS Catalina 图文教程与 macO Catalina.iso 镜像下载地址)

    注 如果需要分享此资源 请注明原作者 不要把别人的东西当做自己的成果 也不要把别人免费分享出来的东西以有偿的方式去分享 自己制作的 macOS Mojave 10 15 3 iso 文件 亲测可用 已在 VMware 15 5 Player
  • AOSP 预置APP安装 以MicroG GmsCore.apk安装为例

    AOSP 预置APP 以MicroG为例 MicroG 无root安装教程 通过源码修改打开签名欺骗 预置不带源码的APP 其他 AOSP 安装谷歌三件套失败 开始寻找代替方案 尝试使用MicroG项目代替谷歌服务 目前情况 已成功安装 所
  • 第十八节 多核异构核间通信–ipcc

    由于MP157 是一款多核异构的芯片 其中既包含的高性能的A7 核及实时性强的M4 内核 那么这两种处理器在工作时 怎么互相协调配合呢 这就涉及到了核间通信的概念了 IPCC inter processor communication co
  • 【医学影像数据处理】2D/3D patch的crop和merge操作汇总

    在做3D分割任务中 多数的方法多采用整体缩放 或裁剪成一个个小的patch操作 这样做的一个主要原因是内存问题 相较于整体缩放 采用裁剪成patch的方法 对于小目标会更加的鲁棒 这也是大多数3D分割任务中常选取的方式 尤其是针对医学影像的
  • spring boot项目同时传递参数和文件的多种方式

    在开发接口中 遇到了需要同时接收参数和文件的情况 可以有多种方式实现文件 参数的接收 这里基于spring boot 3 vue 3 axios 做一个简单的代码演示 1 简单参数 文件参数 参数较少时 比较方便 直接参数接受即可 1 1
  • Mac上亲测好用的BlueStacks蓝叠安卓模拟器

    bluestacks mac是一款基于macOS系统打造的安卓模拟器 其优秀的稳定性 良好的兼容性一直是玩家的模拟器的首选哦 BlueStacks 4 for mac使Mac用户能够在他们的Mac上下载并享受他们喜欢的Android应用程序
  • vs2019运行asp.net framework(c# 基础) 排课系统的完整步骤

    前言 没想过有朝一日还会学net 毕竟java太强了 但因为特殊需要 还是学了一下net 发现她两真的好像呀 源码仓库 https gitee com web paul scheduling system 前期准备 vs2019 需要asp
  • Handler和Controller的区别

    以前一直以为这两个概念貌似是没有太大的区别 调研发现还是有一些区别的 Handler HandlerMapping接口实现从URL映射到请求处理程序bean 支持映射到bean实例和bean names Controller Base Co
  • 2021-08-10基于人脸识别的学生签到系统

    这是这学期web前端开发的大作业 因为要考研时间有限 很多功能只是先把页面做好了 没事实现和数据库的连接 用的数据大多数是在json中存储的 需求分析 一 教师端 功能需求 1登录 数据需求 用户名 密码 邮箱号 2能够管理课程 增加 删除
  • string16类型转string

    std string WChar2Ansi LPCWSTR pwszSrc int nLen WideCharToMultiByte CP ACP 0 pwszSrc 1 NULL 0 NULL NULL if nLen lt 0 retu
  • android 屏幕适配(亲测最兼容方便)

    Android屏幕适配有很多方式 1 设置屏幕固定尺寸适配 例如适配1280x720 只适合少量固定屏幕的设备 2 百分比布局 开发中多了很多代码 3 通过密度值px转dp来适配 方便兼容 基本准确 4 通过修改系统密度值适配 例如抖音适配
  • 【Linux】定时任务crontab/at

    在linux系统中定时任务常用两个命令crontab及at命令 两者区别在于crontab用于设置循环定时任务 即每隔一定时间或固定时间后启动对应任务命令 at用于设置一次性定时任务 在任务完成后定时任务即删除 1 crontab命令 1
  • 【送书活动】借助ChatGPT和Python,轻松实现办公自动化✨

    前言 作者主页 雪碧有白泡泡 个人网站 雪碧的个人网站 推荐专栏 java一站式服务 React从入门到精通 前端炫酷代码分享 从0到英雄 vue成神之路 uniapp 从构建到提升 从0到英雄 vue成神之路 解决算法 一个专栏就够了 架
  • malloc和strcpy,入门的指针面试题

    指针是C和C 编程语言中一个重要的概念 因此在面试以及工作中经常会涉及到指针相关的问题 现在列举几个比较基础问题 一 void getMemory char p p char malloc 100 void test void char s