C语言学习记录——动态内存函数介绍(malloc、free、calloc、realloc)

简介: C语言学习记录——动态内存函数介绍(malloc、free、calloc、realloc)

内存分区

malloc(开辟空间)

函数介绍

malloc是一个申请内存的函数,size表示要申请的内存的空间大小。这个函数的返回值有两种情况,一是在成功申请空间时,返回一个指向这个空间起始地址的void型指针变量;二是当可用内存不足,内存申请失败,则是返回NULL。

函数用法

int arr[10] = {0} //10int等于10 * sizeof(int)
//使用malloc函数来开辟这段空间
int *p = ( int* )malloc(10 * sizeof(int));

free(释放空间)

函数介绍

free函数是用来释放动态开辟的内存。

如果参数memblock指向的空间不是动态开辟的,则这个行为是错误的。(或者说free函数的行为是未定义的)

如果参数memblock是NULL指针,则free函数不起任何作用。

函数用法

结合前面的malloc,来看他们各自的用法。

#include <stdio.h>
#include <stdlib.h>
int main()
{
    //int arr[10] = { 0 };在栈区
    int* p = (int*)malloc(10 * sizeof(int));//动态内存开辟的,在堆区
    //在使用这些内存的时候,判断一下是否开辟成功了
    if (p == NULL)
    {
        //报错
        perror("main");//main:报错内容
        return 0;
    }
    //使用
    int i = 0;
    for (i = 0; i < 10; i++)
    {
        *(p + i) = i;
    }
    //打印一下这个数组
    for (i = 0; i < 10; i++)
    {
        printf("%d ", p[i]);//p[i]等价于 *(p + i)
    }
    //回收空间(释放空间)
    free(p);
    //free释放空间之后不会把指针p赋为空指针
    //如果后面又使用到指针p的话就会造成非法访问内存
    //所以我们需要手动赋成空指针
    p = NULL;
    return 0;
}

运行结果:

如果可用内存不足,内存开辟失败,运行情况则为:

一般地,malloc和free是成对出现的。

calloc(开辟空间)

函数介绍

calloc函数也是用于动态内存分配的,其功能是为num个size大小的元素开辟一块空间,并且把空间的每个字节初始化为0.

与malloc函数主要有两个区别:

  • 函数参数有两个
  • 会进行初始化

函数用法(对比malloc)

realloc(调整空间)

函数介绍

realloc函数的作用是让动态内存管理更加灵活,它可以做到对动态开辟内存大小的调整。

memblock是要调整的内存地址size表示调整之后的新大小

realloc函数同样返回一个void型的指针,指向调整之后的新空间的起始地址

这里 新空间的起始地址有两种情况:

  • 一是原有的空间后面有足够大的空间去给realloc去调整。
  • 二是原有的空间后面没有足够大的空间去给realloc去调整。

而当realloc在堆区已经找不到可以供原有空间进行调整的多余空间时,它就会返回一个NULL指针。


如果我们拿原地址p去接受realloc返回的地址,就有"偷鸡不成蚀把米"的风险,即调整不成功反而把原有的空间丢失了。所以一般地,创建一个临时变量用来存储realloc返回的地址,确认其调整成功之后再赋给p。

函数用法

#include <stdio.h>
#include <stdlib.h>
int main()
{
    //申请了10个int型的空间
    int* p = (int*)calloc(10, sizeof(int));
    int i = 0;
    for (i = 0; i < 10; i++)
    {
        *(p + i) = 6;
    }
    //现在我们要调整为20个int
    int* ptr = (int*)realloc(p, 20 * (sizeof(int)));//创建临时变量ptr
    if (ptr != NULL)//判断是否调整成功
    {
        p = ptr;
    }
    for (i = 10; i < 20; i++)
    {
        *(p + i) = 6;
    }
    for (i = 0; i < 20; i++)
    {
        printf("p[%d] = %d\n",i+1, p[i]);
    }
    free(p);
    p = NULL;
    return 0;
}

运行结果为:

同时,realloc也可以起到malloc的功能

int main()
{
    int* p = (int*) realloc(NULL,40);
    //等价于malloc(40);  
    return 0;
}
目录
打赏
0
1
1
0
74
分享
相关文章
在 C++中,realloc 函数返回 NULL 时,需要手动释放原来的内存吗?
在 C++ 中,当 realloc 函数返回 NULL 时,表示内存重新分配失败,但原内存块仍然有效,因此需要手动释放原来的内存,以避免内存泄漏。
详解C/C++动态内存函数(malloc、free、calloc、realloc)
详解C/C++动态内存函数(malloc、free、calloc、realloc)
1538 1
|
9月前
一刻也没有为它哀悼~接下来登场的是动态内存分配的malloc与realloc以及free函数
一刻也没有为它哀悼~接下来登场的是动态内存分配的malloc与realloc以及free函数
172 0
|
11月前
|
malloc与free的内存管理奥秘:技术分享
【8月更文挑战第22天】在软件开发过程中,内存管理是一个至关重要的环节。特别是在使用C或C++这类语言时,程序员需要手动管理内存的分配与释放。malloc和free函数是这一过程中的核心工具。本文将深入探讨malloc如何分配内存,以及free如何知道释放多少内存,帮助你在工作学习中更好地掌握这一技术干货。
228 4
|
2月前
|
深入理解JVM,包含字节码文件,内存结构,垃圾回收,类的声明周期,类加载器
JVM全称是Java Virtual Machine-Java虚拟机JVM作用:本质上是一个运行在计算机上的程序,职责是运行Java字节码文件,编译为机器码交由计算机运行类的生命周期概述:类的生命周期描述了一个类加载,使用,卸载的整个过类的生命周期阶段:类的声明周期主要分为五个阶段:加载->连接->初始化->使用->卸载,其中连接中分为三个小阶段验证->准备->解析类加载器的定义:JVM提供类加载器给Java程序去获取类和接口字节码数据类加载器的作用:类加载器接受字节码文件。
289 55
Arthas memory(查看 JVM 内存信息)
Arthas memory(查看 JVM 内存信息)
159 6
JVM简介—1.Java内存区域
本文详细介绍了Java虚拟机运行时数据区的各个方面,包括其定义、类型(如程序计数器、Java虚拟机栈、本地方法栈、Java堆、方法区和直接内存)及其作用。文中还探讨了各版本内存区域的变化、直接内存的使用、从线程角度分析Java内存区域、堆与栈的区别、对象创建步骤、对象内存布局及访问定位,并通过实例说明了常见内存溢出问题的原因和表现形式。这些内容帮助开发者深入理解Java内存管理机制,优化应用程序性能并解决潜在的内存问题。
244 29
JVM简介—1.Java内存区域
Elasticsearch集群JVM调优设置合适的堆内存大小
Elasticsearch集群JVM调优设置合适的堆内存大小
1424 1
JVM实战—2.JVM内存设置与对象分配流转
本文详细介绍了JVM内存管理的相关知识,包括:JVM内存划分原理、对象分配与流转、线上系统JVM内存设置、JVM参数优化、问题汇总。
125 12
JVM实战—2.JVM内存设置与对象分配流转
JVM简介—2.垃圾回收器和内存分配策略
本文介绍了Java垃圾回收机制的多个方面,包括垃圾回收概述、对象存活判断、引用类型介绍、垃圾收集算法、垃圾收集器设计、具体垃圾回收器详情、Stop The World现象、内存分配与回收策略、新生代配置演示、内存泄漏和溢出问题以及JDK提供的相关工具。
JVM简介—2.垃圾回收器和内存分配策略
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等

登录插画

登录以查看您的控制台资源

管理云资源
状态一览
快捷访问