学C的第三十二天【动态内存管理】

=========================================================================

相关代码gitee自取:C语言学习日记: 加油努力 (gitee.com)

 =========================================================================

接上期

学C的第三十一天【通讯录的实现】_高高的胖子的博客-CSDN博客

 =========================================================================

                     

1 . 为什么存在动态内存分配

学到现在认识的内存开辟方式有两种:

              

  • 创建变量:       

int val = 20;        ——        在栈空间上开辟4个字节

  • 创建数组:       

char arr[10] = {10};        ——        在栈空间上开辟10个字节的连续空间

                     

                     

                     

上述的开辟空间的方式有两个特点:

               

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

                   

但是对于空间的需求不仅仅是上述的情况

有时候我们需要的空间大小在程序运行的时候才能知道

数组的编译时开辟空间的方式就不能满足了。

这时候就只能试试动态内存开辟了。

         

~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

             

2 . 动态内存函数的介绍

(1). malloc 和 free :

                 

malloc :

该函数是一个动态内存开辟的函数

这个函数可以向内存申请一块连续可用的空间

返回指向这块空间的指针

               

书写格式如下

void* malloc (size_t size); 

                

  • 如果开辟成功,则返回一个指向开辟好空间的指针

                

  • 如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查

                  

  • 返回值的类型void* ,所以malloc函数并不知道开辟空间的类型
  • 具体在使用的时候使用者自己决定

               

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

             

  • malloc声明在 stdlib.h 头文件中。

示例:

                       

free :

malloc函数申请的内存空间程序退出时才会还给操作系统

如果程序不退出,动态申请的内存不会主动释放的。

所以需要 free函数释放动态内存

               

书写格式如下

void free (void* ptr);

                

  • 如果参数 ptr 指向的空间不是动态开辟的,那free函数的行为是未定义的。

        

  • 如果参数 ptr 是NULL指针,则函数什么事都不做

           

  • free声明在 stdlib.h 头文件中。

示例:

                     


                    

(2). calloc :

         

书写格式如下

void* calloc (size_t num, size_t size);

                

  • 函数的功能以 num 个大小为 size 的元素开辟一块空间,并且把空间的每个字节初始化为0

               

  • 如果我们对申请的内存空间的内容要求初始化,那么可以很方便的使用calloc函数来完成任务。

               

  • 与函数 malloc 的区别只在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为全0

              

  • calloc 声明在 stdlib.h 头文件中。

示例:

                     


                    

(3). realloc :

         

realloc函数的出现让动态内存管理更加灵活

有时会我们发现过去申请的空间太小了,有时候我们又会觉得申请的空间过大了,

那为了合理地使用内存,我们一定会对内存的大小做灵活的调整

realloc 函数就可以做到对动态开辟内存大小的调整

                                

书写格式如下

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

                                  

  • ptr 要调整的内存地址,如果填的是NULL空指针,那会开辟一块新的空间跟malloc函数一样

                  

  • size 调整之后新大小

                  

  • 返回值 调整之后的内存起始位置

                  

  • 这个函数调整原内存空间大小的基础上,还会将原来内存中的数据移动到新的空间

                  

  • realloc在调整内存空间的是存在两种情况

情况1 --  原有空间之后足够大的空间:

在这种情况下,要扩展内存直接在原有内存之后直接追加空间

原来空间的数据不发生变化

                           

情况2 --  原有空间之后没有足够大的空间:

在这种情况下,扩展的方法是:在堆空间上另找一个合适大小的连续空间来使用

旧的空间中的数据拷贝到新的空间中,再释放旧的空间,最后返回新空间的起始地址

这样函数返回的就是一个新的内存地址

                  

  • realloc 声明在 stdlib.h 头文件中。

示例:

         

~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

             

3 . 常见的动态内存错误

(1). 对NULL指针的解引用操作:

                      

malloccallocrealloc函数 可能开辟空间
    开辟空间就有可能会失败返回 NULL空指针
    这时解引用该空指针就可能会出问题

示例:

                     


                    

(2). 对动态开辟空间的越界访问:

示例:

                     


                    

(3). 对非动态开辟内存使用free函数释放:

示例:

                     


                    

(4). 使用free函数释放一块动态开辟内存的一部分:

                      

使用动态空间过程中

改变指向动态空间的指针

这时要使用free函数释放空间就会出问题

示例:

                     


                    

(5). 对同一块动态内存多次释放:

                      

可以在释放动态空间后

将该空间指针设置为空指针

防止多次释放

示例:

                     


                    

(6). 动态开辟内存忘记释放 -- 内存泄漏 :

                      

只有两种方式可以对动态内存进行释放

free函数 程序运行结束

所以如果 忘记释放  没释放且程序无法结束

就会造成内存泄漏

示例:

         

~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

             

4 . 相关经典笔试题

题一:

                 

进行修改:

               

对应代码:

//1:改前
#include <stdio.h>
#include <stdlib.h>
#include <string.h>void GetMemory(char* p)
{//开辟动态空间p = (char*)malloc(100);
}void Test(void)
{//创建空指针:char* str = NULL;//使用该指针进行动态内存开辟:GetMemory(str);//对动态空间赋值并使用:strcpy(str, "hello world");printf(str);}int main()
{Test();return 0;
}//1:改后
#include <stdio.h>
#include <stdlib.h>
#include <string.h>void GetMemory(char** p)
{//开辟动态空间*p = (char*)malloc(100);
}void Test(void)
{//创建空指针:char* str = NULL;//使用该指针进行动态内存开辟:GetMemory(&str);//对动态空间赋值并使用:strcpy(str, "hello world");printf(str);//使用后进行释放:free(str);str = NULL;
}int main()
{Test();return 0;
}

                     


                    

题二:

               

进行修改:

               

对应代码:

//2:改前
#include <stdio.h>
#include <stdlib.h>char* GetMemory(void)
{char p[] = "hello world";return p;
}void Test(void)
{//创建空指针:char* str = NULL;//调用上面的函数:str = GetMemory();printf(str);
}int main()
{Test();return 0;
}//2:改后
#include <stdio.h>
#include <stdlib.h>char* GetMemory(void)
{static char p[] = "hello world";return p;
}void Test(void)
{//创建空指针:char* str = NULL;str = GetMemory();printf(str);
}int main()
{Test();return 0;
}

                     


                    

题三:

             

进行修改:

               

对应代码:

//3:改前:
#include <stdio.h>
#include <stdlib.h>void GetMemory(char** p, int num)
{//根据需求创建动态空间:*p = (char*)malloc(num);
}void Test(void)
{//创建空指针变量:char* str = NULL;//调用函数:GetMemory(&str, 100);//使用动态空间:strcpy(str, "hello");printf(str);
}int main()
{Test();return 0;
}//3:改后:
#include <stdio.h>
#include <stdlib.h>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(str);str = NULL;
}int main()
{Test();return 0;
}

                     


                    

题四:

                

进行修改:

               

对应代码:

//4:改前:
#include <stdio.h>
#include <stdlib.h>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;
}//4:改后:
#include <stdio.h>
#include <stdlib.h>void Test(void)
{//创建动态空间并接收:char* str = (char*)malloc(100);//使用动态空间:strcpy(str, "hello");//释放:free(str);str = NULL;if (str != NULL){strcpy(str, "world");printf(str);}
}int main()
{Test();return 0;
}

         

~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

             

5 . C/C++程序的内存开辟

(1). C/C++程序内存区域划分:

                

C/C++程序内存分配的几个区域:

1. 栈区(stack):

执行函数时函数内局部变量的存储单元都可以在栈上创建

函数执行结束时这些存储单元自动被释放

栈内存分配运算内置于处理器的指令集中效率很高,但是分配的内存容量有限

栈区主要存放运行函数而分配的局部变量函数参数返回数据返回地址等。

2. 堆区(heap):

一般由程序员分配释放若程序员不释放

程序结束时可能由OS(操作系统)回收

分配方式类似于链表

3. 数据段(静态区)(static)

存放全局变量静态数据程序结束后由系统释放

4. 代码段:

存放函数体类成员函数和全局函数)的二进制代码

           

图示:

          

 有了这幅图,

我们就可以更好的理解static关键字修饰局部变量的例子了。

实际上普通的局部变量在栈区分配空间的,

栈区的特点在上面创建的变量出了作用域就销毁

但是被static修饰的变量存放在数据段(静态区)

数据段的特点是在上面创建的变量直到程序结束才销毁

所以生命周期变长

         

~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

             

6 . 柔性数组

             

C99中,结构体中最后一个元素允许是未知大小的数组,这就叫做柔性数组成员

              

实例:

                  

(1). 柔性数组的特点:

                    

  • 结构体中的柔性数组成员前面必须至少有一个其他成员

  • sizeof 返回的这种结构体大小不包括柔性数组的内存

  • 包含柔性数组成员的结构体malloc ()函数进行内存的动态分配
    并且分配的内存应该大于结构的大小,以适应柔性数组的预期大小

              

实例:

                     


                    

(2). 柔性数组的使用:

              

实例:

                     


                    

(3). 柔性数组的优势:

方便内存释放

如果我们的代码是在一个给别人用的函数中

在里面做了二次内存分配(使用两次malloc函数可以实现类似柔性数组的效果)

把整个结构体返回给用户

用户调用free可以释放结构体,但是用户并不知道这个结构体内的成员也需要free

所以你不能指望用户来发现这个事

所以,如果我们把结构体的内存以及其成员要的内存一次性分配好了

返回给用户一个结构体指针

用户做一次free就可以把所有的内存也给释放掉

             

有利于访问速度

连续的内存益于提高访问速度

也有益于减少内存碎片(两个开辟的空间中间空余的内存)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/24186.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

mysql数据库备份

数据库备份&#xff0c;数据库为school mysql> create database if not exists school;1.创建student和score表 创建student&#xff1a; CREATE TABLE student ( id INT(10) NOT NULL UNIQUE PRIMARY KEY , name VARCHAR(20) NOT NULL , sex VARCHAR(4) , birth YEAR, d…

C# Blazor 学习笔记(0.1):如何开始Blazor和vs基本设置

文章目录 前言资源推荐环境如何开始Blazor个人推荐设置注释快捷键热重载设置 前言 Blazor简单来说就是微软提供的.NET 前端框架。使用 WebAssembly的“云浏览器”&#xff0c;集成了Vue,React,Angular等知名前端框架的特点。 资源推荐 微软官方文档 Blazor入门基础视频合集 …

vscode无法连接远程服务器的可能原因:远程服务器磁盘爆了

vscode输入密码后一直等待&#xff0c;无法进入远程服务器终端&#xff1a; 同时Remote-SSH输出包含以下内容 在日志中的以下几个部分&#xff1a; [17:15:05.529] > wget download failed 这表明VS Code尝试在远程服务器上下载VS Code服务器时失败了。> Cannot write…

【vue】vue 里面使用 v-html 插入的文本带有换行符‘\n‘不换行

最近开发vue2 项目 &#xff0c;接口返回的是类似于这样的数据&#xff1a;我是第一行的哦\n我是第二行的哦 我是直接这样渲染的&#xff0c; //html <p v-htmltext></p>//渲染值 this.text "我是第一行的哦\n我是第二行的哦"但结果却是不如意&#x…

【JavaScript】本地存储

在JavaScript中&#xff0c;本地存储是一种可以在浏览器中存储数据的机制。它允许开发者在浏览器中保存键值对&#xff0c;并且这些数据可以在同一个域名下的不同页面间进行共享。 JavaScript中常用的本地存储机制有两种&#xff1a;localStorage 和 sessionStorage。 localS…

如何优化Flask应用程序的性能

让我们来谈谈如何优化Flask应用程序的性能。 首先&#xff0c;让我们了解一下什么是性能。性能就是你在做某件事情时所花费的时间和资源。在计算机科学中&#xff0c;性能通常指的是计算机程序运行的速度和质量。对于Web应用程序&#xff0c;性能通常指的是响应时间和服务器的…

js:浏览器环境下普通模式和严格模式use strict下function函数里的this指向

普通模式 普通模式 this 默认指向了Window function foo() {console.log(this); }foo(); // Window严格模式 如果开启严格模式use strict&#xff0c;this 指向的是 undefined function foo() {"use strict";console.log(this); }foo(); // undefined

STM32 UDS Bootloader开发-上位机篇-CANoe制作(3)

文章目录 前言刷写脚本34服务写入数据的实现定时函数writeBlockData函数Checksum总结前言 上一篇文章中介绍了CAPL刷写脚本的大部分内容,本文继续介绍34,36,37服务的实现,及checksum中遇到的坑 刷写脚本 34服务 void requestDownLoad(struct Block hexfile) {gTxBuffer[…

Linux系统部署Python语言开发运行环境

目录 Ubuntu自带python Debian安装python 安装 pip 库列表 安装第三方库 使用国内镜像站 实装 tkinter 库 编写运行代码 测试代码1 1. 创建项目 2. 创建源码文件 3. 写入源代码 4. 修改权限 5. 运行代码 测试代码2 本文的使用环境是Windows的Linux 子系统&…

ChatGPT: 人机交互的未来

ChatGPT: 人机交互的未来 ChatGPT背景ChatGPT的特点ChatGPT的应用场景结论 ChatGPT ChatGPT是一种基于大数据和机器学习的人工智能聊天机器人模型。它由国内团队发明、开发&#xff0c;并被命名为Mental AI。ChatGPT的目标是通过模拟自然对话的方式&#xff0c;提供高效、智能…

Go学习第六天

Golang变量内置pair结构详细说明 变量包括&#xff08;type, value&#xff09;两部分type 包括 static type和concrete type. 简单来说 static type是你在编码是看见的类型(如int、string)&#xff0c;concrete type是runtime系统看见的类型类型断言能否成功&#xff0c;取决…

原型模式(C++)

定义 使用原型实例指定创建对象的种类&#xff0c;然后通过拷贝这些原型来创建新的对象。 应用场景 在软件系统中&#xff0c;经常面临着“某些结构复杂的对象”的创建工作;由于需求的变化&#xff0c;这些对象经常面临着剧烈的变化&#xff0c;但是它们却拥有比较稳定一致的…

Julia 元编程

Julia 把自己的代码表示为语言中的数据结构&#xff0c;这样我们就可以编写操纵程序的程序。 元编程也可以简单理解为编写可以生成代码的代码。 元编程&#xff08;英语&#xff1a;Metaprogramming&#xff09;&#xff0c;是指某类计算机程序的编写&#xff0c;这类计算机程…

Linux磁盘管理

磁盘管理 基本分区管理 磁盘划分思路 进入分区表&#xff0c;新建分区更新分区表格式化分区表挂载使用 #lsblk #df -h 查看设备挂载情况 #fdisk -l 设备分区情况 #fdisk /dev/sdb 添加一块硬盘&#xff0c;需要将其分两个分区&#xff0c;分别格式化成ext4和vfat格式文件系…

Shell脚本学习-for循环结构2

案例&#xff1a;通过脚本实现仅sshd、rsyslog、crond、network、sysstat服务在开机时自启动。 Linux系统在开机的服务通常工作在文本模式3级别&#xff0c;因此只需要查找3级别以上的开启的服务即可。查看命令&#xff1a; chkconfig --list |grep 3:on [rootvm1 ~]# chkco…

安卓camera1设置自动连续对焦

需求 camera 1实现的控制摄像头拍照功能&#xff0e; adb控制自动拍照&#xff0c;发现近点时拍照很模糊&#xff0c;需要自动连续对焦&#xff0e; mCamera.autoFocus(null) 这个接口只能实现单次对焦&#xff0e;不适用&#xff0e; 代码 private Parameters mParameters;p…

TechTool Pro for mac(硬件监测和系统维护工具)

TechTool Pro 是为 Mac OS X 重新设计的全新工具程序&#xff0c;不但保留旧版原有的硬件侦测功能&#xff0c;还可检查系统上其他重要功能&#xff0c;如&#xff1a;网络连接&#xff0c;区域网络等。 TechTool Pro for mac随时监控和保护您的电脑&#xff0c;并可预设定期检…

最新SecureCRT 中文注册版

SecureCRT是一款由VanDyke Software公司开发的终端仿真软件&#xff0c;它提供了类似于Telnet和SSH等协议的远程访问功能。SecureCRT专门为网络管理员、系统管理员和其他需要保密访问网络设备的用户设计。 软件下载&#xff1a;SecureCRT for ma注册版 远程访问&#xff1a;Sec…

MySQL 与MongoDB区别

一、什么是MongoDB呢 ? MongoDB 是由C语言编写的&#xff0c;是一个基于分布式文件存储的开源数据库系统。在高负载的情况下&#xff0c;添加更多的节点&#xff0c;可以保证服务器性能。 MongoDB 旨在为WEB应用提供可扩展的高性能数据存储解决方案。 MongoDB 将数据存储为一…

【PHP代码审计】ctfshow web入门 php特性 93-104

ctfshow web入门 php特性 93-104 web 93web 94web 95web 96web 97web 98web 99web 100web 101web 102web 103web 104 web 93 这段PHP代码是一个简单的源码审计例子&#xff0c;让我们逐步分析它&#xff1a; include("flag.php");: 这行代码将flag.php文件包含进来。…