C语言高频面试题——malloc 和 calloc区别

在 C 语言中,malloccalloc 都是用于动态内存分配的函数,但它们在 内存初始化参数形式使用场景 上有显著区别。以下是详细的对比分析:


1. 函数原型

malloc
c 复制代码
void* malloc(size_t size);
  • 功能 :分配 未初始化 的连续内存块。
  • 参数:需要分配的内存大小(以字节为单位)。
  • 返回值 :指向分配内存的指针,失败时返回 NULL
calloc
c 复制代码
void* calloc(size_t num, size_t size);
  • 功能 :分配 初始化为零 的连续内存块。
  • 参数
    • num:要分配的元素个数。
    • size:每个元素的大小(以字节为单位)。
  • 返回值 :指向分配内存的指针,失败时返回 NULL

2. 核心区别

特性 malloc calloc
初始化 不初始化内存(内容随机) 自动将内存初始化为零
参数形式 单一参数(总字节数) 两个参数(元素个数 + 单个元素大小)
适用场景 需要手动初始化的场景 需要零初始化的场景(如数组、结构体)
性能 更快(无需清零内存) 较慢(需额外时间初始化内存)

3. 初始化差异

malloc 的未初始化内存
  • 分配的内存块内容是随机的(可能包含垃圾值)。

  • 示例:

    c 复制代码
    int* ptr = malloc(5 * sizeof(int));
    if (ptr) {
        printf("%d\n", ptr[0]); // 输出随机值
    }
calloc 的零初始化
  • 分配的内存块会被初始化为零(所有位为 0)。

  • 示例:

    c 复制代码
    int* ptr = calloc(5, sizeof(int));
    if (ptr) {
        printf("%d\n", ptr[0]); // 输出 0
    }

4. 参数形式对比

malloc 的参数
  • 需要手动计算总字节数。

  • 示例:分配 5 个整数的内存:

    c 复制代码
    int* ptr = malloc(5 * sizeof(int)); // 总字节数 = 5 * sizeof(int)
calloc 的参数
  • 直接指定元素个数和单个元素大小,自动计算总字节数。

  • 示例:分配 5 个整数的内存:

    c 复制代码
    int* ptr = calloc(5, sizeof(int)); // 自动计算 5 * sizeof(int)

5. 使用场景

使用 malloc 的情况
  • 需要动态分配内存,但后续会手动初始化。

  • 性能敏感的场景(避免初始化开销)。

  • 示例:

    c 复制代码
    struct Point {
        int x;
        int y;
    };
    struct Point* p = malloc(sizeof(struct Point));
    if (p) {
        p->x = 10; // 手动初始化
        p->y = 20;
    }
使用 calloc 的情况
  • 需要零初始化的场景(如数组、避免未定义行为)。

  • 分配大型数据结构时(如动态数组)。

  • 示例:

    c 复制代码
    int* arr = calloc(100, sizeof(int)); // 分配并初始化为 0
    if (arr) {
        printf("%d\n", arr[50]); // 输出 0
    }

6. 错误处理

无论是 malloc 还是 calloc,都必须检查返回值是否为 NULL

c 复制代码
int* ptr = malloc(1000 * sizeof(int));
if (ptr == NULL) {
    fprintf(stderr, "Memory allocation failed!");
    exit(EXIT_FAILURE);
}

7. 释放内存

两者分配的内存都必须通过 free 释放:

c 复制代码
free(ptr);
ptr = NULL; // 避免悬空指针

总结

  • malloc
    • 快速分配,但需要手动初始化。
    • 适合需要自定义初始化的场景。
  • calloc
    • 自动零初始化,适合数组或需要安全默认值的场景。
    • 分配大块内存时可能稍慢。

选择建议

  • 如果需要零初始化,优先使用 calloc
  • 如果需要手动控制初始化过程,使用 malloc
相关推荐
DashVector1 小时前
如何通过Java SDK分组检索Doc
java·数据库·面试
Code_Artist1 小时前
[Go]结构体实现接口类型静态校验——引用类型和指针之间的关系
后端·面试·go
John.Lewis1 小时前
数据结构初阶(13)排序算法-选择排序(选择排序、堆排序)(动图演示)
c语言·数据结构·排序算法
程序员清风1 小时前
跳表的原理和时间复杂度,为什么还需要字典结构配合?
java·后端·面试
张元清2 小时前
电商 Feeds 流缓存策略:Temu vs 拼多多的技术选择
前端·javascript·面试
Jenny3 小时前
第九篇:卷积神经网络(CNN)与图像处理
后端·面试
前端缘梦3 小时前
深入理解 Vue 中的虚拟 DOM:原理与实战价值
前端·vue.js·面试
天天摸鱼的java工程师3 小时前
Snowflake 雪花算法优缺点(Java老司机实战总结)
java·后端·面试
顾林海4 小时前
Android MMKV 深度解析:原理、实践与源码剖析
android·面试·源码阅读
Java技术小馆4 小时前
重构 Controller 的 7 个黄金法则
java·后端·面试