动态内存分配
目录
- 存在动态内存分配的原因
- 动态内存函数
- malloc
- free
- calloc
- realloc
- 常见的动态内存错误
- C/C++程序的内存开辟
- 柔性数组
- 柔性数组的特点
- 柔性数组的使用
- 柔性数组的优势
存在动态内存分配的原因
内存开辟方式,例如:
int val = 20;
在栈空间上开辟四个字节
char arr[10] = { 0 };
在栈空间上开辟10个字节的连续空间
但是这种开辟空间的方式有两个特点:
- 空间开辟大小是固定的
- 数组在申明的时候,必须指定数组的长度,它所需要的内存在编译时分配
但是有时候我们需要的空间大小,在程序运行的时候才能知道
所以可以试试动态内存开辟
动态内存函数
动态内存开辟在堆区
malloc
void* malloc (size_t size);
size
单位是字节
malloc(40);
申请40个字节的内存块大小
头文件为#include<stdlib.h>
这个函数向内存申请一块连续可用的空间,并返回指向这块空间的指针
- 如果开辟成功,则返回一个指向开辟好空间的指针
malloc
申请到空间后直接返回这块空间的起始地址,不会初始化空间的内容
int i = 0;for (i = 0; i < 10; i++){printf("%d\n", *(p + i));}
打印出来的都是空间的起始地址
- 如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查
if (p == NULL){perror("malloc");return 1;}
在检测到参数(
malloc
)为NULL
后,说明malloc
无法开辟空间,继续进行下去会发生错误,则立刻调用perror
函数,将生成的错误描述将打印出来,后跟一个换行字符('\n'
)
在调用perror
函数后,将程序停止,不再继续进行程序的运行
- 返回值的类型是 void* ,所以malloc函数并不知道开辟空间的类型,具体在使用的时候使用者自己来决定
int* p = (int*)malloc(40);
每次进行
p+1
都会跳过一个int*
类型
- 如果参数 size 为0,malloc的行为是标准是未定义的,取决于编译器
malloc
申请的内存空间,当程序退出时,还给操作系统- 当程序不退出,动态申请的内存,不会主动释放的
- 需要使用
free
函数来释放
free
C语言提供了一个函数free
,专门是用来做动态内存的释放和回收的
void free (void* ptr);
头文件为#include<stdlib.h>
free函数用来释放动态开辟的内存
- 如果参数 ptr 指向的空间不是动态开辟的,那free函数的行为是未定义的
- 如果参数 ptr 是NULL指针,则函数什么事都不做
int main()
{int* p = (int*)malloc(40);//开辟失败if (p == NULL){perror("malloc");return 1;}//开辟成功int i = 0;for (i = 0; i < 10; i++){printf("%d\n", *(p + i));}free(p);p = NULL;return 0;
}
释放掉栈区的p
指针,但是p
仍然指向堆区的空间,但没用了就会成为野指针,所以我们应该主动将p
置为空
calloc
C语言中calloc
函数也用来动态内存分配
void* calloc (size_t num, size_t size);
- 函数的功能是为
num
个大小为size
的元素开辟一块空间,并且把空间的每个字节初始化为0 - 与函数
malloc
的区别只在于calloc
会在返回地址之前把申请的空间的每个字节初始化为全0。 - 头文件为
#include<stdlib.h>
对申请内存空间的内容要求初始化,可以使用calloc
函数来完成
realloc
void* realloc (void* ptr, size_t size);
- ptr 是要调整的内存地址
- size 调整之后新大小(总共的内存大小,包含原来内存空间的大小)
- 返回值为调整之后的内存起始位置。
- 这个函数在调整原内存空间大小的基础上,还会将原来内存中的数据移动到新的空间
realloc
函数的出现让动态内存管理更加灵活。realloc
函数可以做到对动态开辟内存大小的调整
有时会我们发现过去申请的空间太小了,有时候我们又会觉得申请的空间过大了,那为了合理的时候内存,我们一定会对内存的大小做灵活的调整
realloc
在调整内存空间的是存在两种情况
- 原有空间之后有足够大的空间
扩展内存在原有内存之后直接追加空间,原来空间的数据不发生变化
- 原有空间之后没有足够大的空间
在堆空间上另找一个合适大小的连续空间来使用,这样函数返回的是一个新的内存地址,
realloc
内部操作步骤:
- 开辟新的空间
- 会将旧的空间中的数据拷贝到新的空间
- 释放旧的空间
- 返回新空间的起始地址
将起始地址放在一个新的变量中,防止开辟失败将原有空间释放且没有新的空间地址
例如:
int main()
{int* p = (int*)malloc(40);//开辟失败if (p == NULL){perror("malloc");return 1;}//开辟成功 初始化为1-10int i = 0;for (i = 0; i < 10; i++){p[i] = i + 1;}//增加新的空间int* ptr = (int*)realloc(p, 80);if (ptr != NULL){p = ptr;ptr = NULL;}else{perror("realloc");return 1;}//释放空间free(p);p = NULL;return 0;
}
- 增加新的空间时,将增加完空间之后的起始地址放在一个新的指针变量(
ptr
)中- 在确定增加的空间开辟成功后再将新开辟空间的地址(
ptr
)赋给原开辟的指针变量(p
)中- 将新的指针变量(
ptr
)置为NULL,防止最后释放空间后变为野指针- 结束内存空间分配后释放空间(此时
p
和ptr
所处的地址相同,释放p
指针的地址的同时ptr
指针的地址相应的也不存在),置为NULL
常见的动态内存错误
-
对NULL指针的解引用操作
错误示例:
-
对动态开辟空间的越界访问
错误示例:
-
对非动态开辟内存使用free释放
错误示例:
-
使用free释放一块动态开辟内存的一部分
错误示例:
-
对同一块动态内存多次释放
错误示例:
-
动态开辟内存忘记释放(内存泄漏)
错误示例:
忘记释放不再使用的动态开辟的空间会造成内存泄漏
动态申请的内存空间,不会因为出了作用域自动销毁(还给操作系统)
只有两种方式进行销毁:
free
- 程序结束(退出)
C/C++程序的内存开辟
C/C++程序内存分配的几个区域:
- 栈区(
stack
):在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元自动被释放。
栈内存分配运算内置于处理器的指令集中,效率很高,但是分配的内存容量有限。
栈区主要存放运行函数而分配的局部变量、函数参数、返回数据、返回地址等
- 堆区(
heap
):一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收
分配方式类似于链表
- 数据段(静态区)(
static
)存放全局变量、静态数据
程序结束后由系统释放
- 代码段:存放函数体(类成员函数和全局函数)的二进制代码
static
关键字修饰局部变量:
- 普通的局部变量是在栈区分配空间的,栈区的特点是在上面创建的变量出了作用域就销毁
- 但是被static修饰的变量存放在数据段(静态区),数据段的特点是在上面创建的变量,直到程序结束才销毁,所以生命周期变长
柔性数组
在C99 中,结构中的最后一个元素允许是未知大小的数组,这就叫做“柔性数组”成员
typedef struct st_type
{int i;int a[0];//柔性数组成员
}type_a;
若编译器会报错无法编译,也可以写成
typedef struct st_type
{int i;int a[];//柔性数组成员
}type_a;
int a[];
和int a[0]
都成员是不确定的,表示为柔性数组
柔性数组的特点
- 结构中的柔性数组成员前面必须至少有一个其他成员
sizeof
返回的这种结构大小不包括柔性数组的内存
struct S
{int n;int arr[0];
};
int main()
{printf("%d\n", sizeof(struct S));return 0;
}
打印结果为:4
- 包含柔性数组成员的结构用
malloc
函数进行内存的动态分配,并且分配的内存应该大于结构的大小,以适应柔性数组的预期大小
柔性数组的使用
代码1:
struct S
{int n;int arr[0];
};
int main()
{struct S* ps = (struct S*)malloc(sizeof(struct S) + 40);if (ps == NULL){perror("malloc");return 1;}ps->n = 100;int i = 0;for (i = 0; i < 10; i++){ps->arr[i] = i + 1;}//释放free(ps);ps = NULL;return 0;
}
柔性数组成员arr
,相当于获得了40个字节的连续空间
柔性数组的优势
柔性数组也可以用指针来代替
代码2:
struct S
{int n;int* arr;
};
int main()
{struct S* ps = (struct S*)malloc(sizeof(struct S));if (ps == NULL){perror("malloc->ps");return 1;}ps->n = 100;ps->arr = (int*)malloc(40);if (ps->arr == NULL){perror("malloc->arr");return 1;}int i = 0;for (i = 0; i < 10; i++){ps->arr[i] = i + 1;}//释放free(ps->arr);ps->arr = NULL;free(ps);ps = NULL;return 0;
}
代码1 和 代码2 可以完成同样的功能
但是柔性数组的实现有两个好处:
- 方便内存释放
使用柔性数组,结构体的内存以及其成员要的内存一次性就可以分配好,并返回一个结构体指针,做一次free就可以把所有的内存也给释放掉
- 有利于访问速度
连续的内存有益于提高访问速度,也有益于减少内存碎片
相关文章:

动态内存分配
目录 存在动态内存分配的原因动态内存函数mallocfreecallocrealloc 常见的动态内存错误C/C程序的内存开辟柔性数组柔性数组的特点柔性数组的使用柔性数组的优势 存在动态内存分配的原因 内存开辟方式,例如: int val 20;在栈空间上开辟四个字节 char arr[10] { 0 …...

【C语言】常见的动态内存管理错误
前言 上一篇介绍了C语言中 动态内存管理函数,本片讲解的是 在我们使用动态内存管理时 常见的错误,一起来看看吧~ 欢迎关注个人主页:逸狼 创造不易,可以点点赞吗~ 如有错误,欢迎指出~ 目录 1.对NULL指针的解引⽤操作 错…...

数据结构之二叉树的精讲
𝙉𝙞𝙘𝙚!!👏🏻‧✧̣̥̇‧✦👏🏻‧✧̣̥̇‧✦ 👏🏻‧✧̣̥̇:Solitary_walk ⸝⋆ ━━━┓ - 个性标签 - :来于“云”的“羽球人”。…...

ETL是什么
一、ETL概念 ETL,是英文Extract-Transform-Load的缩写,用来描述将数据从来源端经过抽取(extract)、转换(transform)、加载(load)至目的端的过程。ETL一词较常用在数据仓库ÿ…...

华为配置WLAN高密业务示例
配置WLAN高密业务示例 组网图形 图1 配置高密WLAN环境网络部署组网图 业务需求组网需求数据规划配置思路配置注意事项操作步骤配置文件 业务需求 体育场由于需要接入用户数量很大,AP间部署距离较小,因此AP间的干扰较大,可能导致用户上网网…...

C++——类和对象(1)
1. 类 我们之前提及过C语言是面向过程的语言,其解决问题的方式是关注问题过程,然后逐步解决。而C是面向对象编程,聚焦于对象,依靠多个对象之间的交互关系解决问题。而类这个概念的引入则是面向对象的最深刻体现。 1.1 C中的结构体…...

vue+element ui上传图片到七牛云服务器
本来打算做一个全部都是前端完成的资源上传到七牛云的demo,但是需要获取token,经历了九九八十一难,最终还是选择放弃,token从后端获取(springboot)。如果你们有前端直接能解决的麻烦记得私我哦!…...

学不动系列-git-hooks和husky+lintstage
git-hooks 为了保证提交的代码符合规范,可以在上传代码时进行校验。常用husky来协助进行代码提交时的eslint校验。husky是基于git-hooks来实现,在使用husky之前,我们先来研究一下git-hooks。 构建git-hooks测试项目 需要使用git-hooks就需…...
K8S相关小技巧《四》
需求: 我作为Kubernetes的集群管理员,前一段时间有收到一个需求,需要我创建一个受限访问的用户kubeconfig,提供给跳板机的某用户。 该kubeconfig需要在非Kubernetes节点的某跳板机上由指定的非root用户使用,该用户仅能…...

Delphi 报错 Type androidx.collection.ArraySet is defined multiple times
Delphi 11 建立一个新的 Multi-Device Application 编译成app的时候报错 报错信息 [PAClient Error] Error: E7688 Unable to execute "E:\Program\Java\jdk1.8.0_301\bin\java.exe" -cp "e:\program\embarcadero\studio\22.0\bin\Android\r8-3.3.28.jar"…...
Post请求中文乱码问题
url*************************************这里填写自己请求的网址 response requests.post(url, datajson.dumps(body),headersheader) r response.text print 打印乱码长这样: data:{“code”:0,“data”:{“end”:false,“message”:“{\n “ˆ—¡A”: [“…...

LeetCode -- 79.单词搜索
1. 问题描述 给定一个 m x n 二维字符网格 board 和一个字符串单词 word 。如果 word 存在于网格中,返回 true ;否则,返回 false 。 单词必须按照字母顺序,通过相邻的单元格内的字母构成,其中“相邻”单元格是那些水…...

单元测试、集成测试、系统测试有什么不同?
单元测试、集成测试和系统测试是软件测试开发中不可或缺的部分。 单元测试: 范围:单元测试是对软件中最小的可测试单元的测试,通常是函数、方法或类。 目的:它的目标是验证每个单独的单元是否按照预期工作,以增加代码…...

数据迁移DTS | 云上MySQL 数据库迁移至达梦数据库
引入 云上 MySQL 数据库 —> 向达梦国产化数据库迁移 下载&安装 达梦客户端工具 DM->可参考之前国产化专栏达梦文章 创建模式 在客户端分别依次执行以下命令脚本(这里没有通过客户端管理工具去创建达梦数据库的模式,当然也可以通过图形化界…...

Linux进程管理:(二)进程调度原语
文章说明: Linux内核版本:5.0 架构:ARM64 参考资料及图片来源:《奔跑吧Linux内核》 Linux 5.0内核源码注释仓库地址: zhangzihengya/LinuxSourceCode_v5.0_study (github.com) 进程调度的概念比较简单,…...
Compose 介绍
Compose 介绍 Android Compose 是 Google 官方推出的用于构建原生 Android UI 的现代工具包。它使用 Kotlin 语言编写,可以帮助开发人员更轻松、更快速地创建精美、响应式和高性能的 Android 应用。 Compose 的优势 声明式 UI: Compose 使用声明式 UI…...

5分钟搞定Python中函数的参数
函数的灵活性非常高,除了常规定义的位置参数以外,还支持默认参数、关键字参数、以及可变参数 ... 这样以来,不但能应对各种复杂的情况,甚至还可以简化调用者的代码。 位置参数 在调用函数时,一般会根据函数定义的参数…...

Gitlab: 私有化部署
目录 1. 说明 2. 资源要求 3. 安装 4. 配置实践 4.1 服务器 4.2 人员与项目 4.2 部署准备 4.2.1 访问变量及用户账号设置 4.2.2 Runner设置 4.2.3 要点 5. 应用项目 CI/CD 6. 参考 1. 说明 gitlab是一个强大且免费的代码管理/部署工具,能统一集成代码仓…...

深入理解Linux线程(LWP):概念、结构与实现机制(2)
🎬慕斯主页:修仙—别有洞天 ♈️今日夜电波:会いたい—Naomile 1:12━━━━━━️💟──────── 4:59 🔄 ◀️ ⏸ ▶️ ☰ &a…...
VBS脚本搞定,快速批量提取一堆Excel文件中的数据
1.需求诞生 小王就职于一家国有大型企业,工作业务十分繁忙,在处理企业某业务数据时,需要从上千个Excel文件中提取某一单元格位置的数据,并整理到另一个Excel文件。要说是这样的Excel文件仅有几个或者十几个也还好,手动…...
uniapp 对接腾讯云IM群组成员管理(增删改查)
UniApp 实战:腾讯云IM群组成员管理(增删改查) 一、前言 在社交类App开发中,群组成员管理是核心功能之一。本文将基于UniApp框架,结合腾讯云IM SDK,详细讲解如何实现群组成员的增删改查全流程。 权限校验…...

AI-调查研究-01-正念冥想有用吗?对健康的影响及科学指南
点一下关注吧!!!非常感谢!!持续更新!!! 🚀 AI篇持续更新中!(长期更新) 目前2025年06月05日更新到: AI炼丹日志-28 - Aud…...

无法与IP建立连接,未能下载VSCode服务器
如题,在远程连接服务器的时候突然遇到了这个提示。 查阅了一圈,发现是VSCode版本自动更新惹的祸!!! 在VSCode的帮助->关于这里发现前几天VSCode自动更新了,我的版本号变成了1.100.3 才导致了远程连接出…...

智能在线客服平台:数字化时代企业连接用户的 AI 中枢
随着互联网技术的飞速发展,消费者期望能够随时随地与企业进行交流。在线客服平台作为连接企业与客户的重要桥梁,不仅优化了客户体验,还提升了企业的服务效率和市场竞争力。本文将探讨在线客服平台的重要性、技术进展、实际应用,并…...

HBuilderX安装(uni-app和小程序开发)
下载HBuilderX 访问官方网站:https://www.dcloud.io/hbuilderx.html 根据您的操作系统选择合适版本: Windows版(推荐下载标准版) Windows系统安装步骤 运行安装程序: 双击下载的.exe安装文件 如果出现安全提示&…...
AspectJ 在 Android 中的完整使用指南
一、环境配置(Gradle 7.0 适配) 1. 项目级 build.gradle // 注意:沪江插件已停更,推荐官方兼容方案 buildscript {dependencies {classpath org.aspectj:aspectjtools:1.9.9.1 // AspectJ 工具} } 2. 模块级 build.gradle plu…...

10-Oracle 23 ai Vector Search 概述和参数
一、Oracle AI Vector Search 概述 企业和个人都在尝试各种AI,使用客户端或是内部自己搭建集成大模型的终端,加速与大型语言模型(LLM)的结合,同时使用检索增强生成(Retrieval Augmented Generation &#…...
管理学院权限管理系统开发总结
文章目录 🎓 管理学院权限管理系统开发总结 - 现代化Web应用实践之路📝 项目概述🏗️ 技术架构设计后端技术栈前端技术栈 💡 核心功能特性1. 用户管理模块2. 权限管理系统3. 统计报表功能4. 用户体验优化 🗄️ 数据库设…...

Yolov8 目标检测蒸馏学习记录
yolov8系列模型蒸馏基本流程,代码下载:这里本人提交了一个demo:djdll/Yolov8_Distillation: Yolov8轻量化_蒸馏代码实现 在轻量化模型设计中,**知识蒸馏(Knowledge Distillation)**被广泛应用,作为提升模型…...

搭建DNS域名解析服务器(正向解析资源文件)
正向解析资源文件 1)准备工作 服务端及客户端都关闭安全软件 [rootlocalhost ~]# systemctl stop firewalld [rootlocalhost ~]# setenforce 0 2)服务端安装软件:bind 1.配置yum源 [rootlocalhost ~]# cat /etc/yum.repos.d/base.repo [Base…...