这篇文章详细介绍动态内存管理 ,让你醍醐灌顶【c语言】

news/2024/11/17 1:28:10/

文章目录

  • 动态内存函数
    • malloc
    • free
    • calloc
    • realloc
  • 常见的动态内存错误
    • 对NULL指针的解引用操作
    • 对动态开辟空间的越界访问
    • 对非动态开辟内存使用free释放
    • 使用free释放一块动态开辟内存的一部分
    • 对同一块动态内存多次释放
    • 动态开辟内存忘记释放(内存泄漏)
  • 练习
  • 柔性数组
    • 柔性数组的使用
    • 柔性数组的优势

在这里插入图片描述


我们之前的内存开辟方式有局限性 :

空间开辟大小是固定的。
数组在申明的时候,必须指定数组的长度,它所需要的内存在编译时分配。


所以为了解决以上问题,我们需要学习动态内存开辟

动态内存函数

在这里插入图片描述

malloc

在这里插入图片描述

  • malloc函数可以向内存申请一块连续可用的空间,并返回指向这块空间的指针
  • 如果开辟成功,则返回一个指向开辟好空间的指针。
  • 如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查。
  • 返回值的类型是 void* ,所以malloc函数并不知道开辟空间的类型,具体在使用的时候使用者自己
    来决定。
  • 如果参数 size 为0,malloc的行为是标准是未定义的,取决于编译器。
int main()
{int* p = (int*)  malloc(10 * sizeof(int));if (p ==NULL){perror("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));}free(p);//回收空间p = NULL; //  防止指针指向的空间释放,而导致出现野指针  
}

在这里插入图片描述

free

专门是用来做动态内存的释放和回收的

在这里插入图片描述

  • 如果参数 ptr 指向的空间不是动态开辟的,那free函数的行为是未定义的。
int main()
{int a = 10;int* p = &a;free(p);//errreturn 0;
}
  • 如果参数 ptr 是NULL指针,则函数什么事都不做
  • malloc 是申请空间,free是释放空间 ,也就是说malloc 和free 一般都是成对出现

calloc

在这里插入图片描述

  • 函数的功能是为 num 个大小为 size 的元素开辟一块空间,并且把空间的每个字节初始化为0。
  • 与函数 malloc 的区别在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为全0
int main()
{int* p = calloc(10, sizeof(int));if (p == NULL){perror("main");return 1;}int i = 0; for (i = 0; i < 10; i++){printf("%d ", *(p+i) );}free(p);p = NULL;return 0;
}

在这里插入图片描述

realloc

在这里插入图片描述

  • realloc函数的出现让动态内存管理更加灵活。
  • 有时会我们发现过去申请的空间太小了,有时候我们又会觉得申请的空间过大了,那为了合理的时
    候内存,我们一定会对内存的大小做灵活的调整。那 realloc 函数就可以做到对动态开辟内存大小
    的调整
  • realloc 有可能找不到合适的空间来调整大小 ,这时候就返回NULL
int main()
{int* p = (int *)calloc(10, sizeof(int));if (p == NULL)  //开辟失败{perror("main");return 1;}//开辟成功int i = 0;for (i = 0; i < 10; i++){*(p + i) = 5;  }//增容 int * ptr = (int *)realloc(p, 20*sizeof(int));// 增容成功if (ptr != NULL){p = ptr;}free(p); //释放p = NULL;return 0;
}

第一种情况 ,realloc后面空间足够 ,返回原来的空间的数据(A)的地址

在这里插入图片描述

第二种情况 : realloc 后面的空间不足 , realloc 会再开辟一块空间(B) , 将原来的空间的数据(A)拷贝到新开辟的空间(B)处,并且会返回新开辟的空间(B)的首地址 ,同时会将原来的空间(A)还给操作系统 。

在这里插入图片描述

常见的动态内存错误

对NULL指针的解引用操作

int main()
{int*p = (int*)malloc(1000000000);//对malloc函数的返回值,做判断int i = 0;for (i = 0; i < 10; i++){*(p + i) = i;}return 0;
}

malloc 开辟空间可能会失败 ,失败返回NULL,对空指针解引用,会导致非法访问
所以我们一般对malloc函数返回值进行判断

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

int main()
{int* p = (int*)malloc(10 * sizeof(int));if (p == NULL){return 1;}int i = 0;//越界访问for (i = 0; i < 40; i++){*(p + i) = i;}free( p); //释放p = NULL;return 0;
}

malloc 函数只是开辟了10 个整形空间 ,上述代码会出现越界访问

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

int main()
{int arr[10] = { 0 };//栈区int* p = arr;free(p);//使用free释放非动态开辟的空间p = NULL;return 0;
}

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

int main()
{int* p = malloc(10 * sizeof(int));//判断开辟成功if (p == NULL){return 1;}int i = 0;for (i = 0; i < 5; i++){*p++ = i;}//释放free(p);p = NULL;return 0;
}

在这里插入图片描述

free 不完全释放空间
丢失了起始位置的地址,可能找不到这块空间 ,可能会导致内存泄漏

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

int main()
{int* p = (int*)malloc(100);//使用//释放free(p);p = NULL;//...//释放free(p); // 如果参数 ptr 是NULL指针,则free什么事都不做return 0;
}

万一出现多次释放, 只要记得手动置成NULL ,如果参数 ptr 是NULL指针,则free函数什么事都不做,

动态开辟内存忘记释放(内存泄漏)

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

动态开辟空间有两种回收方式 : 通过free函数主动释放 , 整个程序结束

p指向malloc 开辟空间的起始地址 。
p是局部变量,在栈区上开辟空间 , 出作用域就销毁
也就意味着丢失了malloc开辟空间的起始地址 ,就算后续再用free释放也无效
而且malloc开辟的空间并没有销毁,会一直吃内存,导致内存泄漏

练习

void GetMemory(char *p) //p是形参,是str的一份临时拷贝
{
p = (char *)malloc(100);
}
void Test(void)
{
char *str = NULL;
GetMemory(str);
strcpy(str, "hello world");
printf(str);
}
int main()
{Test():return 0 ;
}

在这里插入图片描述

str传给GetMemory函数的时候是值传递,所以GetMemory函数的形参p是str的一份临时拷贝
在GetMemory函数内部动态申请空间的地址,存放在p中,不会影响str
所以GetMemory函数返回之后,str还是NULL 。
所以strcpy会拷贝失败
其次 ,当GetMemory函数返回之后,形参p销毁而且随着p的销毁,malloc创建的空间也找不到了,导致了内存泄漏,后续再使用free也无法释放 。


char *GetMemory(void)
{
char p[] = "hello world";
return p;
}
void Test(void)
{
char *str = NULL;
str = GetMemory();
printf(str);
}

在这里插入图片描述

GetMemory 函数内部创建的数组是在栈区上创建的,p数组进入GetMemory函数生命周期开始,出GetMemory 函数生命周期结束
return p 返回 数组首元素地址(h)
也就意味着str里面存的是h的地址
但是此时p数组里的空间还给操作系统
printf调用str,来打印p数组里的内容 , p数组里的内容很有可能被覆盖 ,(返回的地址是没有实际的意义,如果通过返回的地址,去访问内存就是非法访问内存的)
输出结果自然就是随机值


void GetMemory(char **p, int num)
{
*p = (char *)malloc(num);
}
void Test(void)
{
char *str = NULL;
GetMemory(&str, 100);
strcpy(str, "hello");
printf(str);
}

没有free函数释放malloc开辟的空间,可能会造成内存泄漏

void Test(void)
{
char *str = (char *) malloc(100);
strcpy(str, "hello");
free(str);if(str != NULL){strcpy(str, "world");printf(str);}
}
int main()
{
Test();
return 0 ;
}

在这里插入图片描述

malloc 开辟100个字节的内存空间,由str维护
strcpy函数将hello\0拷贝进malloc开辟的内存空间
strcpy函数将world\0拷贝进malloc开辟的空间 ,会失败
str 虽然记得malloc开辟空间的地址 ,但是此时malloc开辟的空间已经被free释放了,还给操作系统了,此时去访问内存就是非法访问内存的。

柔性数组

  • 结构中的柔性数组成员前面必须至少一个其他成员。
  • sizeof 返回的这种结构大小不包括柔性数组的内存。
struct S
{int n;int arr[0];
};
int main()
{struct S s = { 0 };printf("%d", sizeof(s)); // 输出4 return 0;
}
struct S
{int n;int arr[0];
};
int main()
{struct S s = { 0 };printf("%d", sizeof(s)); //输出4 return 0;
}

上述两种写法 ,看编译器支持哪一种


  • 包含柔性数组成员的结构用malloc ()函数进行内存的动态分配,并且分配的内存应该大于结构的大
    小,以适应柔性数组的预期大小。
  struct S
{int n;//4int arr[0];//大小是未知
};int main(){//期望arr的大小是10个整形struct S* ps = (struct S*)malloc(sizeof(struct S) + 10 * sizeof(int));return 0;}

在这里插入图片描述

柔性数组的使用

  struct S
{int n;//4int arr[0];//大小是未知
};int main(){//期望arr的大小是10个整形struct S* ps = (struct S*)malloc(sizeof(struct S) + 10 * sizeof(int));ps->n = 10;int i = 0;for (i = 0; i < 10; i++){ps->arr[i] = i;}struct S* ptr = (struct S*)realloc(ps, sizeof(struct S) + 20 * sizeof(int));if (ptr != NULL){ps = ptr;}free(ps);ps = NULL;return 0;}

柔性数组的优势

  • 方便内存释放
  • 这样有利于访问速度.

在这里插入图片描述
如果你觉得这篇文章对你有帮助,不妨动动手指给点赞收藏加转发,给鄃鳕一个大大的关注
你们的每一次支持都将转化为我前进的动力!!!


http://www.ppmy.cn/news/7688.html

相关文章

死锁的成因以及解决方案

&#x1f388;专栏链接:多线程相关知识详解 目录 一.什么是死锁以及死锁的成因 Ⅰ.一个线程一把锁 Ⅱ.两个线程两把锁 Ⅲ.多个线程多把锁 二.死锁的解决方案 一.什么是死锁以及死锁的成因 死锁是一个线程加上锁了之后,解不开了 在多线程编程中&#xff0c;我们为了防止多…

剑指offer----C语言版----第二天

目录 1. 二维数组中的查找 1.1 题目描述 1.1 思路一 1.2 思路二 1.3 思路三&#xff08;最优解&#xff09; 1. 二维数组中的查找 原题链接&#xff1a;剑指 Offer 04. 二维数组中的查找 - 力扣&#xff08;LeetCode&#xff09;https://leetcode.cn/problems/er-wei-shu-…

QT学习 控件(二)输入文本类

文章目录QLineEditQTextEditQTextCursorQLineEdit QLineEdit是最基本的输入控件&#xff0c;继承自QObject &#xff0c;常用于短行的输入。 构造函数&#xff1a; 可以指定一个默认文本以及父窗口 QLineEdit(const QString &contents, QWidget *parent nullptr)QLineE…

前端面试题之计算机网络篇--HTTP协议

HTTP协议 1. GET和POST的请求的区别 GET和POST方法 GET和POST方法都是HTTP中的方法 什么是 HTTP&#xff1f; 超文本传输协议&#xff08;Hypertext Transfer Protocol&#xff0c;缩写 HTTP&#xff09;旨在启用客户端和服务器之间的通信。 HTTP 充当客户端和服务器之间的…

花费数小时,带你学透Java数组,这些常用方法你还记得吗?

推荐学习专栏&#xff1a;Java 编程进阶之路【从入门到精通】 文章目录1. 数组2. 一维数组2.1 声明2.2 初始化2.3 使用3. 二维数组3.1 声明3.2 初始化3.3 使用4. 数组在内存中的分布5. 数组常用的方法5.1 Arrays.toString方法5.2 Arrays.copyOf方法5.3 Arrays.copyOfRange方法5…

Qt编写雷达模拟仿真工具1-背景布局

一、前言 雷达模拟仿真工具&#xff0c;整体结构采用的QGraphicsView框架&#xff0c;背景布局采用的分层绘制&#xff0c;这样可以控制该需要重新绘制的重新绘制&#xff0c;不需要重新的绘制的就没必要再多余的浪费&#xff0c;这里定义了一个GraphicsBackGroundItem类继承自…

freeswitch的gateway实现出中继的主备方案

概述 freeswitch是一款简单好用的VOIP开源软交换平台。 某些呼叫场景中&#xff0c;我们有2条出中继线路可选&#xff0c;2条出中继需要按照主备模式来配置&#xff0c;优先使用主中继呼叫&#xff0c;当主中继出现问题时&#xff0c;呼叫自动转移到备用中继呼叫。 本节中&a…

C语言基础 — ( 顺序程序设计[ 运算符、表达式 ] )

欢迎小伙伴的点评✨✨ 本篇章系列是对C语言的深度思考和总结、关于C语言内容会持续更新。 文章目录前言一、C运算符1.1、算数运算符1.2、关系运算符1.3、逻辑运算符1.4、位运算符1.5、赋值运算符1.6、条件运算、指针运算、字节运算1.7、C中的运算优先级二、不同类型数据间的混合…