【C语言基础教程】内存的申请和释放(malloc、free、realloc、calloc)

简介: 【C语言基础教程】内存的申请和释放(malloc、free、realloc、calloc)

前言


C语言是一种强大的编程语言,但它也要求程序员手动管理内存。内存的申请和释放对于有效的资源管理和避免内存泄漏至关重要。在C语言中,有几个关键函数用于内存的动态分配和释放,包括malloc、free、realloc和calloc。本文将介绍为什么C语言需要内存的动态分配和释放,以及这些函数的概念、使用方法和示例代码。


一、为什么需要指针的内存的申请和释放?


在C语言中,内存的静态分配和释放是有限制的,因为你需要在编译时知道变量的大小。然而,许多情况下,我们需要在运行时动态地分配内存,以满足程序的需求。这可以包括在运行时创建数组、字符串、结构等数据结构。这就是动态内存分配和释放的用武之地。


指针的内存的申请和释放允许程序在运行时请求一块内存,使用它,然后在不再需要时释放它,以避免浪费内存和提高程序的效率。以下是四个关键函数,它们使内存的动态管理成为可能。


二、malloc 函数


2.1 函数原型

void* malloc(size_t size);


b1042e47e23d4579893878e1fcf5a703.png

使用方法

malloc 函数用于分配一块指定大小的内存块,并返回一个指向该内存块的指针。你需要提供要分配的内存块的大小(以字节为单位)。如果分配成功,它将返回指向新内存块的指针,否则返回 NULL。


2.2 示例代码 1: 使用 malloc 分配动态数组

#include <stdio.h>
#include <stdlib.h>
int main() {
    int *arr;
    int size = 5;
    arr = (int*)malloc(size * sizeof(int)); // 分配包含5个整数的内存块
    if (arr == NULL) {
        printf("内存分配失败\n");
    } else {
        for (int i = 0; i < size; i++) {
            arr[i] = i * 10;
        }
        for (int i = 0; i < size; i++) {
            printf("%d ", arr[i]);
        }
        free(arr); // 释放内存
    }
    return 0;
}


6c630573aaca45faa2479303964a7719.png

2.3 示例代码 2: 使用 malloc 分配字符串

#include <stdio.h>
#include <stdlib.h>
int main() {
    char *str;
    str = (char*)malloc(50); // 分配包含50个字符的内存块
    if (str == NULL) {
        printf("内存分配失败\n");
    } else {
        strcpy(str, "这是一个动态分配的字符串。");
        printf("%s\n", str);
        free(str); // 释放内存
    }
    return 0;
}


c6f933487a134e1285d1520c873756a3.png


三、free 函数


3.1 函数原型

void free(void* ptr);


7f668e8c8b3849e090f191f9532e02db.png

3.2 使用方法

free 函数用于释放先前使用 malloc、calloc 或 realloc 分配的内存。它接受一个指向要释放的内存块的指针作为参数。成功释放内存后,指针不再指向有效的内存。


3.3 示例代码 1: 使用 free 释放动态数组内存

#include <stdio.h>
#include <stdlib.h>
int main() {
    int *arr;
    int size = 5;
    arr = (int*)malloc(size * sizeof(int)); // 分配内存
    if (arr != NULL) {
        // 使用内存
        for (int i = 0; i < size; i++) {
            arr[i] = i * 10;
        }
        // 释放内存
        free(arr);
    }
    return 0;
}


4607748479f644d3a3717a9add35ed1b.png


3.4 示例代码 2: 使用 free 释放动态分配的字符串

#include <stdio.h>
#include <stdlib.h>
int main() {
    char *str;
    str = (char*)malloc(50); // 分配内存
    if (str != NULL) {
        // 使用内存
        strcpy(str, "这是一个动态分配的字符串。");
        printf("%s\n", str);
        // 释放内存
        free(str);
    }
    return 0;
}


48b2fc63d6344de384b35f7edb8cf73a.png


四、realloc 函数


4.3 函数原型

void* realloc(void* ptr, size_t size);


55d47f8d2b8e417b9d7f66f3b52ed4cf.png

4.2 使用方法

realloc 函数用于更改先前分配的内存块的大小。它接受一个指向先前分配的内存块的指针和新的大小作为参数。如果内存重新分配成功,它将返回一个指向新内存块的指针。如果失败,它将返回NULL,并且原始内存块不会被释放。


4.3 示例代码 1: 使用 realloc 调整动态数组的大小

#include <stdio.h>
#include <stdlib.h>
int main() {
    int *arr;
    int size = 5;
    arr = (int*)malloc(size * sizeof(int)); // 分配内存
    if (arr != NULL) {
        // 使用内存
        // 调整内存大小为10个整数
        arr = (int*)realloc(arr, 10 * sizeof(int));
        if (arr != NULL) {
            for (int i = 0; i < 10; i++) {
                arr[i] = i * 5;
            }
            free(arr); // 释放内存
        }
    }
    return 0;
}


1e335aca883d414d9eb0f18fe541101a.png

4.4 示例代码 2: 使用 realloc 调整动态分配的字符串的大小

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
int main() {
    char *str;
    str = (char*)malloc(50); // 分配内存
  if (str != NULL) {
        // 使用内存
        // 调整内存大小为100个字符
        str = (char*)realloc(str, 100);
        if (str != NULL) {
            strcat(str, " 这是一个追加的字符串。");
            printf("%s\n", str);
            free(str); // 释放内存
        }
    }
        return 0;
}


ef131b4ac3e040eb96c9911d3cfc8bb0.png


五、calloc 函数


5.1 函数原型

void* calloc(size_t numElements, size_t elementSize);


1299f5eb7a1c477caac55997e8be4848.png

5.2 使用方法

calloc 函数用于分配一个包含指定数量元素的内存块,并且每个元素的大小由elementSize参数指定。与malloc不同,calloc分配的内存会被初始化为零。它返回一个指向新内存块的指针。


5.3 示例代码 1: 使用 calloc 分配动态数组并初始化为零

#include <stdio.h>
#include <stdlib.h>
int main() {
    int *arr;
    int size = 5;
    arr = (int*)calloc(size, sizeof(int)); // 分配包含5个整数的内存块并初始化为零
    if (arr != NULL) {
        for (int i = 0; i < size; i++) {
            printf("%d ", arr[i]); // 输出都应该是0
        }
        free(arr); // 释放内存
    }
    return 0;
}


96db8dd1b1d84012916d98b90f8c9167.png

5.4 示例代码 2: 使用 calloc 分配字符串数组并初始化为空

#include <stdio.h>
#include <stdlib.h>
int main() {
    char **strArray;
    int size = 3;
    strArray = (char**)calloc(size, sizeof(char*)); // 分配包含3个字符串指针的数组并初始化为NULL
    if (strArray != NULL) {
        for (int i = 0; i < size; i++) {
            if (strArray[i] == NULL) {
                printf("第%d个字符串为空\n", i + 1);
            }
        }
        free(strArray); // 释放内存
    }
    return 0;
}


358d3eba7f094a5c82a528bf0a92dd96.png


总结

以上是关于内存的申请和释放在C语言中的基本介绍,以及malloc、free、realloc和calloc这四个关键函数的使用示例。通过合理地使用这些函数,你可以动态地分配和释放内存,从而更好地管理资源,避免内存泄漏,并提高程序的效率。在实际编程中,请注意检查内存分配函数的返回值,以确保分配成功,并在不再需要内存时释放它,以避免内存泄漏问题。

相关文章
|
7天前
|
C语言
【C语言】:动态内存管理函数malloc,calloc,realloc和free的介绍的介绍
【C语言】:动态内存管理函数malloc,calloc,realloc和free的介绍的介绍
18 0
|
3天前
|
程序员 C语言
C语言内存管理:malloc、calloc、realloc与free函数详解
C语言内存管理:malloc、calloc、realloc与free函数详解
6 0
|
12天前
|
缓存 C语言
glibc函数malloc的工作原理
glibc函数malloc的工作原理
12 0
|
15天前
|
消息中间件 存储 Kafka
实时计算 Flink版产品使用问题之 从Kafka读取数据,并与两个仅在任务启动时读取一次的维度表进行内连接(inner join)时,如果没有匹配到的数据会被直接丢弃还是会被存储在内存中
实时计算Flink版作为一种强大的流处理和批处理统一的计算框架,广泛应用于各种需要实时数据处理和分析的场景。实时计算Flink版通常结合SQL接口、DataStream API、以及与上下游数据源和存储系统的丰富连接器,提供了一套全面的解决方案,以应对各种实时计算需求。其低延迟、高吞吐、容错性强的特点,使其成为众多企业和组织实时数据处理首选的技术平台。以下是实时计算Flink版的一些典型使用合集。
|
7天前
|
存储 Java C++
Java虚拟机(JVM)管理内存划分为多个区域:程序计数器记录线程执行位置;虚拟机栈存储线程私有数据
Java虚拟机(JVM)管理内存划分为多个区域:程序计数器记录线程执行位置;虚拟机栈存储线程私有数据,如局部变量和操作数;本地方法栈支持native方法;堆存放所有线程的对象实例,由垃圾回收管理;方法区(在Java 8后变为元空间)存储类信息和常量;运行时常量池是方法区一部分,保存符号引用和常量;直接内存非JVM规范定义,手动管理,通过Buffer类使用。Java 8后,永久代被元空间取代,G1成为默认GC。
19 2
|
11天前
|
存储
数据在内存中的存储(2)
数据在内存中的存储(2)
24 5
|
11天前
|
存储 小程序 编译器
数据在内存中的存储(1)
数据在内存中的存储(1)
27 5
|
11天前
|
存储 安全 Java
SpringSecurity6从入门到实战之初始用户如何存储到内存
Spring Security 在 SpringBoot 应用中默认使用 `UserDetailsServiceAutoConfiguration` 类将用户信息存储到内存中。当classpath有`AuthenticationManager`、存在`ObjectPostProcessor`实例且无特定安全bean时,此配置生效。`inMemoryUserDetailsManager()`方法创建内存用户,通过`UserDetails`对象填充`InMemoryUserDetailsManager`的内部map。若要持久化到数据库,需自定义`UserDetailsService`接口实
|
10天前
|
存储 编译器 C语言
数据在内存中的存储
数据在内存中的存储
17 2
|
10天前
|
存储
数据在内存中的存储(了解数据在内存中的存储规则,看这一篇就够了!)
数据在内存中的存储(了解数据在内存中的存储规则,看这一篇就够了!)