<C语言> 动态内存管理
1.动态内存函数
为什么存在动态内存分配?
int main(){int num = 10; //向栈空间申请4个字节int arr[10]; //向栈空间申请了40个字节return 0;
}
上述的开辟空间的方式有两个特点:
- 空间开辟大小是固定的。
- 数组在声明的时候,必须指定数组的长度,它所需要的内存在编译时分配。
但是对于空间的需求,不仅仅是上述的情况。有时候我们需要的空间大小在程序运行的时候才能知道,那数组的编译时开辟空间的方式就不能满足了。这时候就需要动态内存开辟了。
1.1 malloc和free
malloc()
是用于在程序执行期间动态分配内存。它的全称是"memory allocation",意为内存分配。malloc()
函数是C标准库的一部分,它的声明在stdlib.h
头文件中。
函数原型如下:
void* malloc(size_t size);
在这里,size
是你想要分配的字节数,函数返回一个指向分配的内存块起始地址的指针。malloc()
函数的返回类型是void*
,这意味着返回的指针可以赋值给任何指针类型而无需显式转换。
下面简要解释一下malloc()
的工作原理:
1.你提供想要分配的字节数,
malloc()
在堆内存中搜索一个足够大的连续内存块来存储这些字节。2.如果找到了合适的内存块,它将其标记为已使用,并返回该内存块的起始地址的指针。
3.如果找不到足够大的内存块,它将返回一个
NULL
指针,表示内存分配失败。
注意:使用malloc()
分配的内存需要使用free()
函数显式地释放,否则会导致内存泄漏。
void free(void* ptr);
free()
函数接受之前分配的内存块的指针,并将其释放,使其可供将来的动态分配使用。如果忘记释放之前分配的内存,程序每次运行分配代码时都会消耗更多内存,最终可能导致内存耗尽。
#include <stdio.h>
#include <stdlib.h>int main() {int n = 5;int* dynamicArray = (int*)malloc(n * sizeof(int));if (dynamicArray == NULL) {printf("内存分配失败!\n");} else {// 使用分配的内存块for (int i = 0; i < n; i++) {dynamicArray[i] = i + 1;}// 当不再需要分配的内存时,记得释放它free(dynamicArray);dynamicArray = NULL; }return 0;
}
在调用free()
函数释放动态分配的内存后,将指针dynamicArray
设置为NULL
是一个良好的习惯,但不是必须的。
设置指针为NULL
的优点:
- 避免悬挂指针(Dangling Pointer):如果在释放内存后不将指针设置为
NULL
,该指针将仍然保留先前的地址。如果你在后续代码中继续使用该指针,可能会导致悬挂指针,即指针指向的内存已经被释放,这可能导致程序崩溃或产生难以调试的错误。将指针设置为NULL
可以帮助你避免这种情况,因为如果尝试使用空指针,程序将产生明确的错误(空指针解引用)。- 避免重复释放:在释放内存后,如果将指针设置为
NULL
,你可以通过检查指针是否为NULL
来确定是否已经释放了内存。如果你在后续代码中错误地再次调用free()
,会导致未定义的行为。
如果你在后续代码中小心地避免悬挂指针和重复释放内存,那么不设置为NULL
也不会导致问题。然而,这是一个简单且有助于防范错误的额外保护措施,所以建议在释放内存后将指针设置为NULL
。
1.2 calloc
calloc()
是另一个动态内存分配函数,也属于标准C库(stdlib.h头文件)。与malloc()
功能类似,但在使用上有一些区别。
calloc()
函数的原型如下:
void* calloc (size_t num, size_t size);
其中num
是你想要分配的元素数量,size
是每个元素的大小(以字节为单位)。calloc()
函数会为num * size
字节的内存块分配空间,并将该内存块中的所有位初始化为零。
相对于malloc()
,calloc()
的一个优势是它会自动初始化分配的内存,这意味着你不需要手动将分配的内存清零。在某些情况下,这可能是非常有用的,特别是当你需要确保分配的内存一开始就是零值时。
实例:
#include <stdio.h>
#include <stdlib.h>int main() {int n = 5;int* dynamicArray = (int*)calloc(n, sizeof(int));if (dynamicArray == NULL) {printf("内存分配失败!\n");} else {// 使用分配的内存块,这里的内存已经被初始化为零for (int i = 0; i < n; i++) {printf("%d ", dynamicArray[i]); // 输出: 0 0 0 0 0}// 当不再需要分配的内存时,记得释放它free(dynamicArray);}return 0;
}
总结:
calloc = malloc+memset 初始化为0
1.3 realloc
realloc
是一个用于重新分配内存块大小的函数。具体而言,它可以用于更改之前通过malloc
或calloc
分配的内存块的大小。
realloc
函数的声明如下:
void *realloc(void *ptr, size_t size);
参数说明:
ptr
:指向之前已分配内存块的指针。如果ptr
为NULL,则realloc
的行为就相当于malloc
,即分配一个新的内存块。size
:新的内存块大小,以字节为单位。
realloc
的工作原理如下:
- 如果
ptr
为NULL,那么realloc
的行为就等同于malloc(size)
,它将分配一个新的大小为size
字节的内存块,并返回指向该内存块的指针。 - 如果
size
为0,且ptr
不为NULL,那么realloc
的行为就等同于free(ptr)
,即释放掉之前分配的内存块,并返回NULL指针。 - 如果
ptr
不为NULL且size
不为0,realloc
将尝试重新分配之前分配的内存块。可能发生以下几种情况:- 如果之前分配的内存块大小大于或等于
size
,则不会分配新的内存块,而是简单地返回原始内存块的指针,不会改变原内存块的内容。 - 如果之前分配的内存块大小小于
size
,realloc
会尝试将原始内存块扩展到新的大小。这可能会**在原始内存块后面的可用内存空间进行扩展,如果没有足够的连续空间来扩展,则realloc
可能会在另一个地方重新分配一个新的内存块,并将原始内容复制到新的内存块中。**这意味着realloc
有可能返回一个新的指针,而不是原始指针,所以在使用realloc
后,最好将返回的指针赋值给原来的指针。 - 如果
realloc
在新的内存块分配失败时,将返回NULL,并且之前分配的内存块仍然保持未更改。
- 如果之前分配的内存块大小大于或等于
使用realloc
时,应该特别注意以下几点:
- 如果
realloc
返回NULL,表示重新分配失败,原来的指针仍然有效,为避免内存泄漏,应该保存原来的指针,并根据需要释放之前的内存块。 - 当使用
realloc
时,最好不要直接修改原始指针,而是将realloc
的结果赋值给原始指针,以防止意外的内存问题。
实例:
#include <stdio.h>
#include <stdlib.h>
int main() {int *p = (int *) malloc(40);if (p == NULL)return 1;//使用int i = 0;for (i = 0; i < 10; i++) {*(p + i) = i;}for (i = 0; i < 10; i++) {printf("%d ", *(p + i));}//增加空间// p = (int *)realloc(p, 80); //如果开辟失败的话,p变成了空指针,不能这么写int *ptr = (int *) realloc(p, 80);if (ptr != NULL) {p = ptr;ptr = NULL;}//当realloc开辟失败的时候,返回的也是空指针//使用for (i = 10; i < 20; i++) {*(p + i) = i;}for (i = 10; i < 20; i++) {printf("%d ", *(p + i));}//释放free(p);p = NULL;return 0;
}//输出结果:0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
2.常见的动态内存错误
2.1 对NULL指针的解引用操作
#include <stdio.h>
#include <stdlib.h>
int main() {int* p = (int*)malloc(20);*p = 5; //错误,空指针解引用//为了不对空指针解引用 需要进行判断if (p == NULL) {perror("malloc");return 1;}else {*p = 5;}free(p);p = NULL;return 0;
}
2.2 对动态开辟空间的越界访问
#include <stdio.h>
#include <stdlib.h>
int main() {int *p = (int *) malloc(20);if (p == NULL)return 1;int i = 0;for (i = 0; i < 20; i++)//越界访问 20个字节 只能访问5个整型{*(p + i) = i;}free(p);p = NULL;return 0;
}
2.3 对非动态开辟内存使用free释放
#include <stdio.h>
#include <stdlib.h>
int main() {int a = 10;int* p = &a;free(p);// ok?return 0;
}
编译器会直接报错
2.4 使用free释放一块动态开辟内存的一部分
#include <stdio.h>
#include <stdlib.h>
int main() {int *p = (int *) malloc(40);if (p = NULL)return 1;int i = 0;for (i = 0; i < 5; i++) {*p = i;p++;}//释放//在释放的时候,p指向的不再是动态内存空间的起始位置free(p);// p不再指向动态内存的起始位置p++;return 0;
}
2.5 对同一块动态内存多次释放
#include <stdio.h>
#include <stdlib.h>
int main() {int* p = (int*)malloc(40);if (p == NULL)return 1;int i = 0;for (i = 0; i < 5; i++) {*(p + i) = i;}//重复freefree(p);p = NULL;//如果将p赋值为NULL 就可以在free,否则编译器会直接报错free(p);return 0;
}
2.6 动态开辟内存忘记释放(内存泄漏)
#include <stdio.h>
#include <stdlib.h>
int *get_memory() {int *p = (int *) malloc(40);return p;
}int main() {int *ptr = get_memory();//使用//释放 如果不释放 就会导致内存泄漏free(ptr);return 0;
}
3.C/C++程序的内存开辟
C/C++程序内存分配的几个区域:
- 栈区(stack):在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元自动被释放。栈内存分配运算内置于处理器的指令集中,效率很高,但是分配的内存容量有限。 栈区主要存放运行函数而分配的局部变量、函数参数、返回数据、返回地址等。
- 堆区(heap):一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收 。分配方式类似于链表。
- 数据段(静态区)(static)存放全局变量、静态数据。程序结束后由系统释放。
- 代码段:存放函数体(类成员函数和全局函数)的二进制代码。
普通的局部变量是在栈区分配空间的,栈区的特点是在上面创建的变量出了作用域就销毁。 但是被static修饰的变量存放在数据段(静态区),数据段的特点是在上面创建的变量,直到程序结束才销毁,所以生命周期变长。
4.经典笔试题
4.1 题目1
#include <stdio.h>
#include <stdlib.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;
}
请问运行Test 函数会有什么样的结果?
运行Test
函数会导致未定义行为。
在GetMemory
函数中,传入的char *p
是一个局部变量,当在函数内部对其进行修改,并不会影响到原始调用函数中的指针。这是因为函数的参数是通过值传递的,即函数得到的是实参的副本,对参数的修改不会影响原始的实参。
在Test
函数中,将一个NULL指针str
传递给GetMemory
函数,然后在GetMemory
函数中分配了内存并将新的地址赋给p
。但这对str
并没有影响,str
仍然是一个NULL指针,指向未分配的内存。
接着,在Test
函数中使用strcpy
将字符串拷贝到str
指向的内存,但是str
指向的内存并没有被分配,这将导致未定义行为。
为了正确地分配内存并使用指针,需要修改GetMemory
函数,使其返回分配的内存地址,并在Test
函数中接收返回的指针。另外,别忘了在使用完内存后,需要使用free
函数来释放动态分配的内存。
改写1:
#include <stdio.h>
#include <stdlib.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() {char *p = (char *) malloc(100);return p;
}void Test(void) {char *str = NULL;str = GetMemory(); //接受返回的pstrcpy(str, "hello world");printf(str);//释放free(str);str = NULL;
}int main() {Test();return 0;
}
4.2 题目2
char *GetMemory(void) {char p[] = "hello world";return p;
}void Test(void) {char *str = NULL;str = GetMemory();printf(str);
}
请问运行Test函数会有什么样的结果?
在GetMemory
函数中,定义了一个局部数组char p[] = "hello world";
,然后将该数组的地址返回给调用者。但是,一旦GetMemory
函数执行完毕,其局部变量(p
数组)将被销毁,因为它是一个自动存储类别的局部变量。所以,返回的指针指向的是已经无效的内存。
在Test
函数中,你将GetMemory
的返回值赋给指针str
,然后使用printf
打印str
指向的内容。由于GetMemory
返回的是一个无效的指针(指向已经被销毁的局部数组),printf
可能会打印出垃圾值,或者程序崩溃,或者导致其他不可预测的结果。
这个问题被称为"悬挂指针"问题,因为指针悬挂在指向已经无效的内存位置上。
要解决这个问题,可以考虑使用动态内存分配来分配存储字符串的内存,并在使用完后记得使用free
来释放内存。
修改后的代码示例:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>char *GetMemory(void) {char *p = (char *)malloc(strlen("hello world") + 1);if (p != NULL) {strcpy(p, "hello world");}return p;
}void Test(void) {char *str = NULL;str = GetMemory();if (str != NULL) {printf("%s\n", str);free(str); // 释放内存}
}int main() {Test();return 0;
}
4.3 题目3
void GetMemory(char **p, int num) {*p = (char *) malloc(num);
}void Test(void) {char *str = NULL;GetMemory(&str, 100);strcpy(str, "hello");printf(str);
}
请问运行Test函数会有什么样的结果?
没有释放内存,导致内存泄漏
修改后的代码实例:
void GetMemory(char **p, int num) {*p = (char *)malloc(num);
}void Test(void) {char *str = NULL;GetMemory(&str, 100);if (str != NULL) {strcpy(str, "hello");printf("%s\n", str);free(str); // 释放内存}
}
4.4 题目4
void Test(void) {char *str = (char *) malloc(100);strcpy(str, "hello");free(str);if (str != NULL) {strcpy(str, "world");printf(str);}
}
请问运行Test 函数会有什么样的结果?
str被提前释放,再次访问str会导致野指针行为
相关文章:

<C语言> 动态内存管理
1.动态内存函数 为什么存在动态内存分配? int main(){int num 10; //向栈空间申请4个字节int arr[10]; //向栈空间申请了40个字节return 0; }上述的开辟空间的方式有两个特点: 空间开辟大小是固定的。数组在声明的时候,必须指定数组的…...

【ASPICE】:学习记录
学习记录 ASPICE中文资料什么是ASPICE过程参考模型 ASPICE全称“Automotive Software Process Improvement and Capability dEtermination”,即“汽车软件过程改进及能力评定”模型框架 ASPICE中文资料 主要资料来源 什么是ASPICE 过程参考模型...
图论--最短路问题
图论–最短路问题 邻接表 /* e[idx]:存储点的编号 w[idx]:存储边的距离(权重) */ void add(int a, int b, int c) {e[idx] b;ne[idx] h[a];w[idx] ch[a] idx ; }1.拓扑排序 给定一个 n 个点 m 条边的有向图,点的编号是 11 到 n…...

go 结构体 - 值类型、引用类型 - 结构体转json类型 - 指针类型的种类 - 结构体方法 - 继承 - 多态(interface接口) - 练习
目录 一、结构体 1、python 与 go面向对象的实现: 2、初用GO中的结构体:(实例化一个值类型的数据(结构体)) 输出结果不同的三种方式 3、实例化一个引用类型的数据(结构体) 4、…...

盘点16个.Net开源项目
今天一起盘点下,16个.Net开源项目,有博客、商城、WPF和WinForm控件、企业框架等。(点击标题,查看详情) 一、一套包含16个开源WPF组件的套件 项目简介 这是基于WPF开发的,为开发人员提供了一组方便使用自…...

记录对 require.js 的理解
目录 一、使用 require.js 主要是为了解决这两个问题二、require.js 的加载三、main.js 一、使用 require.js 主要是为了解决这两个问题 实现 js 文件的异步加载,避免网页失去响应;管理模块之间的依赖性,便于代码的编写和维护。 二、require.…...

minio-分布式文件存储系统
minio-分布式文件存储系统 minio的简介 MinIO基于Apache License v2.0开源协议的对象存储服务,可以做为云存储的解决方案用来保存海量的图片,视频,文档。由于采用Golang实现,服务端可以工作在Windows,Linux, OS X和FreeBSD上。配置…...

Kindling the Darkness: A Practical Low-light Image Enhancer论文阅读笔记
这是ACMMM2019的一篇有监督暗图增强的论文,KinD其网络结构如下图所示: 首先是一个分解网络分解出R和L分量,然后有Restoration-Net和Adjustment-Net分别去对R分量和L分量进一步处理,最终将处理好的R分量和L分量融合回去。这倒是很常…...
AcWing 4575. Bi数和Phi数
文章目录 题意:思路:代码 题意: 就是给你n个数,对于每一个数y你都需要找到一个最小x使得 ϕ ( x ) ≥ y \phi(x) \ge y ϕ(x)≥y,然后再求一个最小平和。 思路: 其实最开始以来的思路就是二分,我先进行线性筛求出每个数的欧拉函数…...

《Federated Unlearning via Active Forgetting》论文精读
文章目录 1、概述2、方法实验主要贡献框架概述 3、实验结果比较方法实验结果忘却完整性忘却效率模型实用性 4、总结 原文链接: Federated Unlearning via Active Forgetting 1、概述 对机器学习模型隐私的⽇益关注催化了对机器学习的探索,即消除训练数…...

Java课题笔记~Maven基础知识
一、什么是Maven? Maven是专门用于管理和构建Java项目的工具。 它的主要功能有: 提供了一套标准化的项目结构提供了一套标准化的构建流程(编译,测试,打包,发布……)提供了一套依赖管理机制 …...

xcode中如何显示文件后缀
xcode14.3 用不惯mac电脑真恶心,改个显示文件后缀找半天 1、首先双击打开xcode软件 2、此时,电脑左上角出现xcode字样(左上角如果看不到xcode字样,再次点击xcode软件弹出来就有了),鼠标右键它,点击setting或者Prefere…...

SpringBoot使用JKS或PKCS12证书实现https
SpringBoot使用JKS或PKCS12证书实现https 生成JKS类型的证书 可以利用jdk自带的keytool工具来生成证书文件, 默认生成的是JKS证书 cmd命令如下: 执行如下命令,并按提示填写证书内容,最后会生成server.keystore文件 keytool -genkey tomcat…...

云原生势不可挡,如何跳离云原生深水区?
云原生是云计算领域一大热词,伴随云原生概念而来的是数字产业迎来井喷、数字变革来临、数字化得以破局以及新一波的技术红利等等。云原生即“云”原生,顾名思义是让“应用”最大程度地利用云的能力,发挥云价值的最佳路径。具体来说࿰…...

python的decimal或者叫Decimal,BigDecimal
前言 在python中进行小数计算时,很容易发生精度错误问题!!!!一定要注意!!!或者说,只要进行小数的运算都要用decimal。如:银企对账;工程计算等等在…...
Mac环境变量问题
查询环境变量 echo $PATH 查询当前使用的Shell,这里注意SHELL需要大写 echo $SHELL >>>如果输出的是/bin/zsh,说明使用的是zsh。zsh读取的个人配置文件是~/.zshrc (mac10.15.x 后对应的是~/.zprofile) >>>如果输出的是/bin/bash&…...
Shell脚本学习-Web服务监控
参考我的博客文章《Centos安装nginx》,先来安装下nginx。我按照该文档操作了一遍,还是很快就能安装好nginx的。 确认可以安装成功: [rootvm1 sbin]# netstat -atunlp |grep 80 tcp 0 0 0.0.0.0:80 0.0.0.0:* …...

【ChatGPT】基于WSL+Docker的ChatGPT PLUS共享服务部署
最近买了ChatGPT PLUS服务,想通过web服务将它共享给其他人使用,搜了一下目前GitHub上比较热门的服务有 ChatGPT-Next-Webchatgpt-web-share 其中chatgpt-web-share支持API和PLUS账号分享两种方式,且架构为PythonJSDocker,相对比…...

【论文阅读24】Better Few-Shot Text Classification with Pre-trained Language Model
论文相关 论文标题:Label prompt for multi-label text classification(基于预训练模型对少样本进行文本分类) 发表时间:2021 领域:多标签文本分类 发表期刊:ICANN(顶级会议) 相关代…...
119、Spring容器启动流程是怎样的(配有Spring启动完整流程图)
Spring容器启动流程是怎样的 在创建Spring容器,也就是启动Spring时:首先会进行扫描,扫描得到所有的BeanDefinition对象,并存在一个Map中然后筛选出非懒加载的单例BeanDefinition进行创建Bean,对于多例Bean不需要在启动…...
变量 varablie 声明- Rust 变量 let mut 声明与 C/C++ 变量声明对比分析
一、变量声明设计:let 与 mut 的哲学解析 Rust 采用 let 声明变量并通过 mut 显式标记可变性,这种设计体现了语言的核心哲学。以下是深度解析: 1.1 设计理念剖析 安全优先原则:默认不可变强制开发者明确声明意图 let x 5; …...
五年级数学知识边界总结思考-下册
目录 一、背景二、过程1.观察物体小学五年级下册“观察物体”知识点详解:由来、作用与意义**一、知识点核心内容****二、知识点的由来:从生活实践到数学抽象****三、知识的作用:解决实际问题的工具****四、学习的意义:培养核心素养…...

Java-41 深入浅出 Spring - 声明式事务的支持 事务配置 XML模式 XML+注解模式
点一下关注吧!!!非常感谢!!持续更新!!! 🚀 AI篇持续更新中!(长期更新) 目前2025年06月05日更新到: AI炼丹日志-28 - Aud…...
C++ 基础特性深度解析
目录 引言 一、命名空间(namespace) C 中的命名空间 与 C 语言的对比 二、缺省参数 C 中的缺省参数 与 C 语言的对比 三、引用(reference) C 中的引用 与 C 语言的对比 四、inline(内联函数…...
C++.OpenGL (10/64)基础光照(Basic Lighting)
基础光照(Basic Lighting) 冯氏光照模型(Phong Lighting Model) #mermaid-svg-GLdskXwWINxNGHso {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-GLdskXwWINxNGHso .error-icon{fill:#552222;}#mermaid-svg-GLd…...
大模型多显卡多服务器并行计算方法与实践指南
一、分布式训练概述 大规模语言模型的训练通常需要分布式计算技术,以解决单机资源不足的问题。分布式训练主要分为两种模式: 数据并行:将数据分片到不同设备,每个设备拥有完整的模型副本 模型并行:将模型分割到不同设备,每个设备处理部分模型计算 现代大模型训练通常结合…...

OPENCV形态学基础之二腐蚀
一.腐蚀的原理 (图1) 数学表达式:dst(x,y) erode(src(x,y)) min(x,y)src(xx,yy) 腐蚀也是图像形态学的基本功能之一,腐蚀跟膨胀属于反向操作,膨胀是把图像图像变大,而腐蚀就是把图像变小。腐蚀后的图像变小变暗淡。 腐蚀…...

JVM 内存结构 详解
内存结构 运行时数据区: Java虚拟机在运行Java程序过程中管理的内存区域。 程序计数器: 线程私有,程序控制流的指示器,分支、循环、跳转、异常处理、线程恢复等基础功能都依赖这个计数器完成。 每个线程都有一个程序计数…...

深入浅出深度学习基础:从感知机到全连接神经网络的核心原理与应用
文章目录 前言一、感知机 (Perceptron)1.1 基础介绍1.1.1 感知机是什么?1.1.2 感知机的工作原理 1.2 感知机的简单应用:基本逻辑门1.2.1 逻辑与 (Logic AND)1.2.2 逻辑或 (Logic OR)1.2.3 逻辑与非 (Logic NAND) 1.3 感知机的实现1.3.1 简单实现 (基于阈…...
【学习笔记】erase 删除顺序迭代器后迭代器失效的解决方案
目录 使用 erase 返回值继续迭代使用索引进行遍历 我们知道类似 vector 的顺序迭代器被删除后,迭代器会失效,因为顺序迭代器在内存中是连续存储的,元素删除后,后续元素会前移。 但一些场景中,我们又需要在执行删除操作…...