详细谈谈C语言中动态内存

目录
  • 前言
  • 1.关于动态内存的函数
    • 1.1malloc和free函数
    • 1.2calloc函数
    • 1.3realloc函数
  • 2.常见的动态内存错误
    • 2.1对NULL指针解引用
    • 2.2对动态内存开辟的空间越界访问
    • 2.3 对非动态开辟内存使用free释放
    • 2.4 使用free释放一块动态开辟内存的一部分
    • 2.5对同一块动态内存多次释放
    • 2.6内存泄漏
  • 补充:为什么要引入动态内存分配
  • 总结

前言

关于动态内存管理,可能有学习过的小伙伴,也有没有听说过的。没有听说过的小伙伴会觉得很奇怪啊,为什么要动态开辟内存,内存怎么还能是动态的。给不知道的小伙伴解释一下。咱们平时在内存开辟空间也就是在栈区(局部变量,函数参数等等),静态区(全局变量,static修饰的局部变量等等)开辟空间。只能是用多少开辟多少,是非常局限的。有时候我们需要的空间大小在程序运行的时候才能知道,那数组的编译时开辟空间的方式就不能满足了,所以就试试动态内存存储。而动态内存开辟的空间都是在内存中的堆空间的。

1.关于动态内存的函数

1.1 malloc和free函数

malloc这个函数向内存申请一块连续可用的空间,并返回指向这块空间的指针,如果开辟成功,则返回一个指向开辟好空间的指针;如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查。

free函数专门是用来做动态内存的释放和回收的。如果参数 ptr 指向的空间不是动态开辟的,那free函数的行为是未定义的;如果参数 ptr 是NULL指针,则函数什么事都不做。

说明:

该函数设计的巧妙的一点就是返回值为void*,因为函数创造者并不知道使用者是想以什么样的类型指针来接收动态开辟的空间,所以使用者在使用时只需要强行转换成自己想要的类型就可以:

int* p = (int*)malloc(40);//假设是整型时的例子

在使用完动态开辟的空间后记得要用free函数向操作系统释放开辟的空间,并且将指针赋为空指针:

int* p = (int*)malloc(40);

//......

free(p);//避免内存泄漏
p = NULL;//原指针指向的位置既然已经还给操作系统的,就将指针赋为空指针,避免野指针等问题

如果参数 size 为0,malloc的行为是标准是未定义的,取决于编译器。

需要引头文件stdlib.h

1.2 calloc函数

calloc可与malloc函数相对照来学习,calloc函数的功能是为 num 个大小为 size 的元素开辟一块空间,并且把空间的每个字节初始化为0。与malloc有两点不同:

参数不同,calloc需要指明开辟空间的数据类型和该类型数据的个数,而malloc是全部字节数;

calloc在开辟空间的同时将空间内的全部字节初始化为0.

1.3 realloc函数

realloc函数用来调整开辟空间的大小,有时可能开辟的空间小了,有时候可能开辟的大了,都可以通过realloc函数来修改,并且会将原来内存中的数据移动到新的空间。

realloc在调整空间大小时有两种情况:

原有空间后有足够大的空间,这种情况下扩展内存就直接原有内存之后直接追加空间,原来空间的数据不发生变化,返回值为原来位置的指针;

 2.原有空间后没有足够大的空间,这时便在堆空间找到大小合适的连续空间使用,并将原有空间的数据转移到新的空间返回值为新空间的地址。

int main()
{
    int *str = (int*)malloc(10);
    if(str != NULL)
{
    //.....
}
else
{
    exit(EXIT_FAILURE);
}
    //扩展容量
//代码1
str = (int*)realloc(str, 1000);//这样可以吗?(如果申请失败会返回一个空指针!!!)

//修改代码2
int*p = NULL;
p = realloc(str, 1000);
if(p != NULL)
{
    str = p;
}
//...
free(str);
return 0;
}

2.常见的动态内存错误

2.1 对NULL指针解引用

int* p = (int*)malloc(40);
*p=10;
free(p);

这就是忽略了返回值可能是空指针的可能,如果开辟失败会返回空指针。所以最好的做法就是判断一下指针是否为空,然后再进行下一步。

int* p = (int*)malloc(40);
if(p != NULL)
{
    *p=10;
}
free(p);

2.2 对动态内存开辟的空间越界访问

就类似于使用数组时对数组越界访问。

int i = 0;
int *p = (int *)malloc(10*sizeof(int));
if(NULL == p)
{
    exit(EXIT_FAILURE);
}
for(i=0; i<=10; i++)
{
    *(p+i) = i;//当i是10的时候越界访问
}
free(p);

2.3 对非动态开辟内存使用free释放

int a = 0;
int* p = &a;
free(p);

通过查阅MSDN可以发现:Attempting to free an invalid pointer (a pointer to a memory block that was not allocated by calloc, malloc, or realloc) may affect subsequent allocation requests and cause errors. 就是说如果用free释放非calloc,malloc,realloc函数动态开辟的空间可能会导致后续的分配请求并且导致错误。

2.4 使用free释放一块动态开辟内存的一部分

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

会导致动态开辟的空间无法完全释放,进而可能会导致内存泄漏。

2.5 对同一块动态内存多次释放

int *p = (int *)malloc(10);
free(p);
free(p);//重复释放

2.6 内存泄漏

void test()
{
    int *p = (int *)malloc(100);
    if(NULL != p)
    {
        *p = 20;
    }
}
int main()
{
    test();
    return 0;
}

如果使用完动态内存又不释放则会导致这块内存无法在后续被利用,导致内存泄漏。动态开辟的空间一定要释放,并且正确释放。

补充:为什么要引入动态内存分配

1.指针只能指向一个确切的内存空间,欲使用一个数据,必须先设定一个变量来保存它么?

2.在程序设计时,数据多为动态的。即程序运行时数据项的数量是变化的。

3.用数组保存多个元素时,很难预知实际运行时存储的元素个数,往往会导致预定义的元素个数不足或过多

例如:电话簿的管理程序。当添加新联系人时,数据项将增加;删除联系人时,数据项将减少。

动态数据结构可以在运行时灵活地添加、删除或重排数据项。
动态内存管理可以在运行时分配更多的内存空间或释放掉不再需要的空间,因而可以优化存储空间的使用。

所以———由于无法预知在运行时数组元素的使用情况,在程序中预定义的数组大小,如果过小,会导致程序运行失败;如果过大,则会浪费内存空间。

如果在运行时根据实际需要来决定内存的使用情况,则可以很好的解决以上问题。

总结

到此这篇关于C语言中动态内存的文章就介绍到这了,更多相关C语言动态内存内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C语言 动态内存分配详解

    C语言 动态内存分配详解 动态内存分配涉及到堆栈的概念:堆栈是两种数据结构.堆栈都是数据项按序排列的数据结构,只能在一端(称为栈顶(top))对数据项进行插入和删除. 栈(操作系统):由操作系统自动分配释放 ,存放函数的参数值,局部变量的值等.其操作方式类似于数据结构中的栈. 堆(操作系统): 一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收,分配方式倒是类似于链表. \在C语言中,全局变量分配在内存中的静态存储区,非静态的局部变量(包括形参)是分配在内存的动态存储区,该存储区被

  • C语言动态内存分配函数的实现

    在C中我们开辟内存空间有两种方式 : 1.静态开辟内存 :例如: int a;int b[10]; 这种开辟内存空间的特点是 所开辟的内存是在栈中开辟的固定大小的 ,如a是4字节 ,数组b是40字节 ,并且数组在申明时必须指定其长度 , 如果是全局数组的话,内存是在编译时分配好的,如果是局部变量数组的话,运行时在栈上静态分配内存.不管是全局数组还是局部数组,它们都有一个特点,那就是数组大小是确定的,是代码中写死的.那如果我们想在程序运行时才确定一个数组的大小 , 前两种在栈上分配内存的方法显然是

  • C语言动态内存分配的详解

    C语言动态内存分配的详解 1.为什么使用动态内存分配 数组在使用的时候可能造成内存浪费,使用动态内存分配可以解决这个问题. 2. malloc和free C函数库提供了两个函数,malloc和free,分别用于执行动态内存分配和释放. (1)void *malloc(size_t size); malloc的参数就是需要分配的内存字节数.malloc分配一块连续的内存.如果操作系统无法向malloc提供更多的内存,malloc就返回一个NULL指针. (2)void free(void *poi

  • C语言 动态内存分配的详解及实例

    1. 动态内存分配的意义 (1)C 语言中的一切操作都是基于内存的. (2)变量和数组都是内存的别名. ①内存分配由编译器在编译期间决定 ②定义数组的时候必须指定数组长度 ③数组长度是在编译期就必须确定的 (3)但是程序运行的过程中,可能需要使用一些额外的内存空间 2. malloc 和 free 函数 (1)malloc 和 free 用于执行动态内存分配的释放 (2)malloc 所分配的是一块连续的内存 (3)malloc 以字节为单位,并且返回值不带任何的类型信息:void* mallo

  • C语言中关于动态内存分配的详解

    目录 一.malloc 与free函数 二.calloc 三.realloc 四.常见的动态内存的错误 [C语言]动态内存分配 本期,我们将讲解malloc.calloc.realloc以及free函数. 这是个动态内存分配函数的头文件都是 <stdlib.h>. c语言中动态分配内存的函数,可能有些初学c语言的人不免要问了:我们为什么要通过函数来实现动态分配内存呢? 首先让我们熟悉一下计算机的内存吧!在计算机的系统中大致有这四个内存区域: 1)栈:在栈里面储存一些我们定义的局部变量以及形参(

  • C语言动态内存管理的实现

    目录 1. 摘要 2. 为什么存在动态内存管理 3. 动态内存函数 3.1 malloc 3.2 free 3.3 calloc 3.4 realloc 4. 常见的动态内存错误 5. 几个经典笔试题 参考答案 6. 参考文献 1. 摘要 本文主要详解C语言中的动态内存分配 2. 为什么存在动态内存管理 我们先来看一段变量的声明: double x = 1.000000; char str[] = "abcdef"; 好的,上述变量的声明有何特点呢? 请思考一下,我的朋友. 对,没错,

  • c语言动态内存分配知识点及实例

    c语言怎么实现动态内存分配 我们经常会预先给程序开辟好内存空间,然后进行操作. int arr[5] ; 对这个数组我们在定义的时候必须给提前开辟好空间,并且在程序执行的过程中,这个开辟的内存空间是一直存在的,除非等到这个函数执行完毕,才会将空间释放.有个问题就是这个数组在程序中无法被修改. 这些问题给我们造成了一些使用上的不方便,所以,C中提供了malloc()函数. 关于malloc()函数,这个函数它接受一个参数:就是所需的内存的字节数.然后malloc()找到可用内存中那一个大小适合的块

  • 详细谈谈C语言中动态内存

    目录 前言 1.关于动态内存的函数 1.1malloc和free函数 1.2calloc函数 1.3realloc函数 2.常见的动态内存错误 2.1对NULL指针解引用 2.2对动态内存开辟的空间越界访问 2.3 对非动态开辟内存使用free释放 2.4 使用free释放一块动态开辟内存的一部分 2.5对同一块动态内存多次释放 2.6内存泄漏 补充:为什么要引入动态内存分配 总结 前言 关于动态内存管理,可能有学习过的小伙伴,也有没有听说过的.没有听说过的小伙伴会觉得很奇怪啊,为什么要动态开辟

  • 详解C语言中动态内存管理及柔性数组的使用

    目录 一.malloc 二.free 三.calloc 四.realloc 1.realloc在扩容时的情况 2.realloc也能实现malloc功能 五.使用动态内存的常见错误 1.free空指针 2.对动态开辟的空间越界访问 3.对非动态开辟内容free 4.只free动态开辟空间的一部分 5.对同一块内存多次free 6.动态内存空间忘记释放(内存泄漏) 六.柔性数组 1.柔性数组的概念 2.柔性数组的特点 3.柔性数组的使用场景 4.柔性数组的优点 一.malloc 这个函数向堆区申请

  • C语言中动态内存分配malloc、calloc和realloc函数解析

    目录 前言 free函数 malloc函数 calloc函数 realloc函数 扩充 malloc/calloc/realloc区别总结 总结 前言 有时候我们需要的空间大小不确定,需要随着程序需要的空间而变化, 那以数组开辟的固定大小的空间就不适用了, 这时候我们就需要动态分配开辟空间了.当空间不够时就扩容.动态开辟是在堆区开辟一块连续可用空间,并返回这块空间的地址.有三种函数malloc, calloc和realloc.我们动态内存分配就在堆区开辟空间 上面的四个区只有堆区的空间是需要手动

  • C语言中动态内存管理图文详解

    目录 1.动态内存开辟的原因 2.动态内存函数的介绍 2.1malloc和free 2.2calloc 2.3realloc 3.常见的动态内存错误 3.1对NULL指针的解引用操作 3.2对动态开辟空间的越界访问 3.3对非动态开辟内存使用free 3.4使用释放一块动态开辟内存的一部分 3.5对同一块动态内存多次释放 3.6动态开辟内存忘记释放(内存泄漏) 4.练习 4.1练习1 4.1练习2 4.3练习3 4.4练习4 5.C/C++程序的内存开辟 总结 1.动态内存开辟的原因 常见的内存

  • 一文详解C++中动态内存管理

    目录 前言 1.C/C++程序的内存开辟 2.C语言中动态内存管理方式:malloc/calloc/realloc/free 2.1malloc.calloc.realloc区别? 3.C++内存管理方式 3.1 new/delete操作内置类型 3.2 new和delete操作自定义类型 3.3new和malloc处理失败 4.operator new与operator delete函数 4.1 operator new与operator delete函数 4.1.1 我们看看operator

  • C语言编程动态内存开辟实现升级版通讯录教程示例

    目录 前言 一.存放联系人信息 二.通讯录初始化 三.增加联系人 四.销毁通讯录 后记 前言 所谓动态内存开辟的通讯录,就是我需要多少联系人,就给多少联系人,防止给定一个联系人上限,需要增加联系人无法扩容,而联系人没有上限那么多又会造成内存浪费. 本文继之前的静态通讯录作出改进,有兴趣的同学可以看看之前的文章:C语言实现静态通讯录 一.存放联系人信息 这里是用struct PeoInfodata结构体指针指向通讯录,而不再直接 struct PeoInfo data[Max] 用结构体数组定义通

  • 新手向超详细的C语言实现动态顺序表

    目录 一.各个函数接口的实现 1.1 不太好''李姐''的"容量检测函数" 1.2 在任意位置插入的函数"坑!" 1.3 在任意位置删除数据的函数 1.4 其余简单的接口函数 二.顺序表结构体声明与定义 三.头文件的调用 一.各个函数接口的实现 1.1 不太好''李姐''的"容量检测函数" 对顺序表进行插入数据时,需要判断顺序表的容量是否充足,增加数据的同时需要反复地检测容量,所以推荐直接将以上步骤封装成一个函数. 函数实现算法:若容量大小 ==

  • C语言初识动态内存管理malloc calloc realloc free函数

    目录 一.为什么存在动态内存分配 二.动态内存函数的使用 1.malloc函数 (1)malloc的定义 (2)malloc函数的注意事项 (3)malloc函数的使用 2.calloc函数 (1)calloc函数的定义 (2)calloc函数的注意事项 (3)calloc函数的使用 3.realloc函数 (1)realloc函数的定义 (2)realloc函数的注意事项 (3)realloc函数的使用 总结 一.为什么存在动态内存分配 在c语言中我们目前掌握的内存开辟方式有: int val

  • 深入了解C语言的动态内存管理

    目录 一.为什么会存在动态内存 二.动态内存函数 1.malloc和free 2.calloc 3.realloc 三.动态内存函数常见错误 2.对NULL指针进行解引用操作 3.使用free释放一块动态开辟内存的一部分 4.对静态内存进行free释放 5.对同一内存空间多次释放 6.动态开辟空间忘记释放 四.经典笔试题 1.笔试1 2.笔试2 3.笔试3 总结 一.为什么会存在动态内存 int data=20;//在栈空间上开辟4个字节空间 char ch[5]={0};//在栈开辟5个字节连

  • Go语言中节省内存技巧方法示例

    目录 引言 预先分配切片 结果 结构体中的字段顺序 极端情况 使用 map[string]struct{} 而不是 map[string]bool 结果 引言 GO虽然不消耗大量内存,但是仍有一些小技巧可以节省内存,良好的编码习惯是每一个程序员都应该具备的素质. 预先分配切片 数组是具有连续内存的相同类型的集合.数组类型定义时要指定长度和元素类型. 因为数组的长度是它们类型的一部分,数组的主要问题是它们大小固定,不能调整. 与数组类型不同,切片类型无需指定长度.切片的声明方式与数组相同,但没有数

随机推荐