当前位置: 首页 > news >正文

C语言:动态内存管理

在这里插入图片描述

文章目录

  • 一、动态内存函数
    • 1. malloc
    • 2. calloc
    • 3. realloc
    • 4. free
  • 二、常见的错误
    • 1.malloc或calloc开辟的空间未检查
    • 2.越界访问
    • 3.对非malloc和calloc开辟的空间,用free释放
    • 4.对同一块动态内存多次释放
    • 5.用free释放动态内存的一部分
  • 三、通讯录(动态版本改写)
  • 总结


一、动态内存函数

要介绍动态内存函数,我们就要知道动态内存函数开辟的空间在内存的什么位置。
如下图所示:
动态内存函数开辟的空间在堆区
动态内存函数开辟的空间在堆区,该区域的空间,只有free函数和程序结束才会释放。

1. malloc

函数的声明:void* malloc (size_t size);包含在<stdlib.h>头文件中。
调用该函数可以向内存中申请大小为size个字节的连续空间,并返回该空间的起始地址。
注意:

  • 如果malloc开辟空间成功,则返回成功开辟好空间的起始地址
  • 如果malloc开辟空间失败,则返回NULL
  • 不要开辟大小为0的空间,这是标准未定义的。

因此,malloc开辟空间后,我们要检查返回值。

#include <stdio.h>
#include <stdlib.h>int main()
{int* pa = (int*)malloc(sizeof(int) * 10);//开辟大小为40个字节的空间//检查是否空间开辟失败if (pa == NULL){exit(-1);}free(pa);pa = NULL;return 0;
}

2. calloc

函数的声明:void* calloc (size_t num, size_t size);包含在<stdlib.h>头文件中。
调用该函数可以向内存中申请num个大小是size个字节的连续空间,再将空间每一个字节初始化为0,并返回该空间的起始地址。
calloc与malloc的唯一区别在于,calloc会将空间每一个字节初始化为0。
如下所示:

#include <stdio.h>
#include <stdlib.h>int main()
{int* arr1 = (int*)malloc(sizeof(int) * 10);if (arr1 == NULL){exit(-1);}for (int i = 0; i < 10; i++){printf("%d ", arr1[i]);}printf("\n");int* arr2 = (int*)calloc(sizeof(int), 10);if (arr2 == NULL){exit(-1);}for (int i = 0; i < 10; i++){printf("%d ", arr2[i]);}free(arr1);free(arr2);arr1 = NULL;arr2 = NULL;return 0;
}

在这里插入图片描述

3. realloc

函数的声明:void* realloc (void* ptr, size_t size);包含在<stdlib.h>头文件中。
调用该函数可以调整由malloc和calloc开辟的空间大小,调整为size个字节大小的空间。
但要注意,其调整空间大小的方式有两种,原地扩容和异地扩容。

  • 原地扩容,原地址空间后空间大小充足,在原地址空间直接扩容。

  • 异地扩容,原地址空间后空间大小不够,在堆区另找一片充足的空间使用。

  • 如果调整的大小过大,realloc会返回NULL。

//异地扩容
int main()
{int* pa = (int*)malloc(sizeof(int) * 10);if (pa == NULL){exit(-1);}printf("%p\n", pa);int* tmp = (int*)realloc(pa, sizeof(int) * 20);if (tmp != NULL){pa = tmp;printf("%p\n", pa);}free(pa);pa = NULL;return 0;
}

在这里插入图片描述


//原地扩容
int main()
{char* pa = (char*)malloc(sizeof(char) * 10);if (pa == NULL){exit(-1);}printf("%p\n", pa);char* tmp = (char*)realloc(pa, sizeof(char) * 20);if (tmp != NULL){pa = tmp;printf("%p\n", pa);}free(pa);pa = NULL;return 0;
}

在这里插入图片描述

4. free

函数的声明:void free (void* ptr);包含在<stdlib.h>头文件中。
调用该函数可以释放由malloc和calloc开辟的空间,指针变量ptr本身内容不变,需要程序员本身置空。

  • 如果ptr是NULL,则函数什么事都不会做
  • 如果ptr并不是malloc和calloc开辟的空间,那free函数的行为未定义

如下:

int main()
{int* pa = (int*)malloc(sizeof(int) * 10);if (pa == NULL){exit(-1);}free(pa);//释放pa所指向的空间pa = NULL;//pa置NULL,防止野指针return 0;
}

二、常见的错误

1.malloc或calloc开辟的空间未检查

有时开辟空间过大,而未检查时,会发生对NULL的解引用


int main()
{int* pa = (int*)malloc(sizeof(int) * INT_MAX);*pa = 10;free(pa);pa = NULL;return 0;
}

在这里插入图片描述

2.越界访问

访问超出开辟空间大小

int main()
{int* pa = (int*)malloc(sizeof(int) * 10);if (pa == NULL){exit(-1);}for (int i = 0; i < 11; i++){pa[i] = 0;}free(pa);pa = NULL;return 0;
}

在这里插入图片描述

3.对非malloc和calloc开辟的空间,用free释放

free函数只能用了释放动态内存空间

int main()
{int arr[10] = { 0 };free(arr);return 0;
}

在这里插入图片描述

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

动态内存空间释放一次即可

int main()
{int* pa = (int*)malloc(sizeof(int) * 10);if (pa == NULL){exit(-1);}free(pa);free(pa);pa = NULL;return 0;
}

在这里插入图片描述

5.用free释放动态内存的一部分

开辟的动态内存空间实际大小要大于我们申请的大小,那多出的一部分空间要记录我们这次申请空间的信息,free函数就可以根据这一部分信息来释放我们申请的空间大小。如果我们传递的指针并未指向空间首地址,那么free就找不到信息来释放空间。

int main()
{int* pa = (int*)malloc(sizeof(int) * 10);if (pa == NULL){exit(-1);}pa++;free(pa);pa = NULL;return 0;
}

在这里插入图片描述

三、通讯录(动态版本改写)

静态版本通讯录
相对于静态版本而言,动态版本改变并不多也不难,下面就是要改变的部分。

  1. 结构
//动态版本
typedef struct Contact
{PeoInfo* data;int sz;//记录此时已用的大小int capacity;//记录通讯录的大小
}Contact;

2.初始化InitContact

#define INITSIZE 10//初始化通讯录
void InitContact(Contact* pc)
{pc->sz = 0;pc->capacity = INITSIZE;pc->data = (PeoInfo*)malloc(sizeof(PeoInfo) * INITSIZE);if (pc->data == NULL){perror("malloc");return;}
}
  1. 扩容AddCapacity
bool AddCapacity(Contact* pc)
{PeoInfo* tmp = realloc(pc->data, sizeof(PeoInfo) * (pc->capacity) * 2);if (tmp == NULL){perror("realloc");return false;}pc->data = tmp;pc->capacity *= 2;return true;
}//添加联系人
void AddContact(Contact* pc)
{assert(pc);if (pc->sz == pc->capacity){if (AddCapacity(pc) == true){printf("扩容成功\n");}else{printf("扩容失败\n");return;}}printf("请输入联系人名字:>");scanf("%s", pc->data[pc->sz].name);printf("请输入联系人性别:>");scanf("%s", pc->data[pc->sz].sex);printf("请输入联系人年龄:>");scanf("%d", &(pc->data[pc->sz].age));printf("请输入联系人电话:>");scanf("%s", pc->data[pc->sz].tele);printf("请输入联系人地址:>");scanf("%s", pc->data[pc->sz].adder);pc->sz += 1;
}
  1. 退出ExitContact
//退出通讯录
void ExitContact(Contact* pc)
{free(pc->data);pc->data = NULL;pc->sz = 0;pc->capacity = 0;
}

总结

以上就是我对于动态内存管理的了解和应用。

相关文章:

C语言:动态内存管理

文章目录 一、动态内存函数1. malloc2. calloc3. realloc4. free 二、常见的错误1.malloc或calloc开辟的空间未检查2.越界访问3.对非malloc和calloc开辟的空间&#xff0c;用free释放4.对同一块动态内存多次释放5.用free释放动态内存的一部分 三、通讯录(动态版本改写)总结 一、…...

如何往MySQL中插入100万条数据?

需求 现在有一个 数据量 为100万的数据样本 100w_data.sql 其数据格式如下&#xff0c;截取最后十条数据 999991,XxGdnLZObA999991,XxGdnLZObA,XxGdnLZObA,2020-3-18,1 999992,TBBchSKobC999992,TBBchSKobC,TBBchSKobC,2020-9-8,2 999993,rfwgLkYhUz999993,rfwgLkYhUz,rfwgLk…...

IntelliJ IDEA 2023.2 最新变化

主要更新 AI Assistant 限定访问 Ultimate 在此版本中&#xff0c;我们为 IntelliJ IDEA 引入了一项重要补充 – AI Assistant。 AI Assistant 当前具备一组由 AI 提供支持的初始功能&#xff0c;提供集成式 AI 聊天&#xff0c;可以完成一些任务&#xff0c;例如自动编写文档…...

1300*B. T-primes

解析&#xff1a; 有且只有三个因数&#xff0c;当且仅当&#xff0c;完全平方数并且sqrt&#xff08;n&#xff09;为素数 #include<bits/stdc.h> using namespace std; typedef long long ll; const int N1e55; ll t,n; bool prime(ll x){if(x<2) return 0;for(int…...

重新C++系列之运算符重载

一、什么是运算符重载 简单来讲就是对运算符赋予新的意义&#xff0c;但是又不能改变原有的含义&#xff0c;它本身也就是一个函数。运算符重载的本质是以函数的方式来体现。 二、运算符重载有几种 1、按照作用域来划分&#xff0c;有全局操作符重载函数和成员函数操作符重载函…...

kotlin异常处理try-catch-finally

kotlin异常处理try-catch-finally fun main(args: Array<String>) {try {println("a")} catch (e: Exception) {//异常捕获println("a-catch: $e")} finally {//善后&#xff0c;无论是否异常&#xff0c;都会执行println("a-finally")}t…...

Pytorch在cuda、AMD DirectML和AMD CPU下性能比较

一、测试环境 CUDA环境: i7-8550u 16G DDR4 2133MHz nVidia MX150 2GB AMD DirectML环境: Ryzen 5 5600G 32G DDR4 3200MHz Vega7 4GB AMD 纯CPU环境&#xff1a;Ryzen 5 5600G 32G DDR4 3200MHz 其他硬件配置的硬盘、电源均一致。Pytorch版本为2.0.0&#xff0c;Pyt…...

哈工大计算机网络课程局域网详解之:交换机概念

哈工大计算机网络课程局域网详解之&#xff1a;交换机概念 文章目录 哈工大计算机网络课程局域网详解之&#xff1a;交换机概念以太网交换机&#xff08;switch&#xff09;交换机&#xff1a;多端口间同时传输交换机转发表&#xff1a;交换表交换机&#xff1a;自学习交换机互…...

Jenkins Pipeline的hasProperty函数

函数的作用 用于判断某个参数或者字段是否存在。 用法 例子一 def projectStr "P1,P2,P3" pipeline {agent anyparameters {extendedChoice(defaultValue: "${projectStr}",description: 选择要发布的项目,multiSelectDelimiter: ,,name: SELECT_PROJ…...

芯片制造详解.净洁室的秘密.学习笔记(三)

这是芯片制造系列的第三期跟学up主三圈&#xff0c;这里对其视频内容做了一下整理和归纳&#xff0c;喜欢的可以看原视频。 芯片制造详解03&#xff1a; 洁净室的秘密&#xff5c;为何芯片厂缺人&#xff1f; 芯片制造详解.净洁室的秘密.学习笔记 三 简介一、干净的级别二、芯片…...

可解释的 AI:在transformer中可视化注意力

Visualizing Attention in Transformers | Generative AI (medium.com) 一、说明 在本文中&#xff0c;我们将探讨可视化变压器架构核心区别特征的最流行的工具之一&#xff1a;注意力机制。继续阅读以了解有关BertViz的更多信息&#xff0c;以及如何将此注意力可视化工具整合到…...

k8s Webhook 使用java springboot实现webhook 学习总结

k8s Webhook 使用java springboot实现webhook 学习总结 大纲 基础概念准入控制器&#xff08;Admission Controllers&#xff09;ValidatingWebhookConfiguration 与 MutatingWebhookConfiguration准入检查&#xff08;AdmissionReview&#xff09;使用Springboot实现k8s-Web…...

JS逆向之猿人学爬虫第20题-wasm

文章目录 题目地址sign参数分析python算法还原往期逆向文章推荐题目地址 https://match.yuanrenxue.cn/match/20第20题被置顶到了第1页,题目难度 写的是中等 算法很简单,就一个标准的md5算法,主要是盐值不确定, 而盐值就在wasm里面,可以说难点就在于wasm分析 sign参数分…...

【双指针优化DP】The 2022 Hangzhou Normal U Summer Trials H

Problem - H - Codeforces 题意&#xff1a; 思路&#xff1a; 首先很明显是DP 因为只有1e6个站点&#xff0c;因此可以以站点作为阶段 注意到K很小&#xff0c;因此可以尝试把这个当作第二维 设dp[i][j]为到达第i个站点&#xff0c;已经花了j元钱的最小步数 然后就想了一…...

[论文笔记] LLM数据集——金融数据集

一、chatglm_金融 ModelScope 魔搭社区 请将modelscope sdk升级到v1.7.2rc0&#xff0c;执行&#xff1a; ​ pip3 install "modelscope1.7.2rc0" -f https://modelscope.oss-cn-beijing.aliyuncs.com/releases/repo.html # 方式1 git clone http://www.modelscope…...

在亚马逊平台,如何有效举报违规行为?

众所周知&#xff0c;在每个行业都有一些违规现象&#xff0c;甚至这些违规现象还会给自己带来利益方面的损失&#xff0c;一旦触犯到自己的利益的话&#xff0c;那自己是需要想办法解决的&#xff0c;想办法规避。 就拿开亚马逊店铺来说&#xff0c;比较容易遇到的就是产品侵…...

深度学习入门教学——神经网络

深度学习就是训练神经网络。 1、神经网络 举个最简单的例子&#xff0c;以下是一个使用线性回归来预测房屋价格的函数。这样一个用于预测房屋价格的函数被称作是一单个神经元。大一点的神经网络&#xff0c;就是将这些单个神经元叠加起来。例如&#xff1a;神经网络根据多个相…...

阿里Java开发手册~OOP 规约

1. 【强制】避免通过一个类的对象引用访问此类的静态变量或静态方法&#xff0c;无谓增加编译器解析成 本&#xff0c;直接用 类名 来访问即可。 2. 【强制】所有的覆写方法&#xff0c;必须加 Override 注解。 说明&#xff1a; getObject() 与 get 0 bject() 的问题。…...

【Mysql数据库面试01】内连接 左连接 右连接 全连接

【Mysql数据库】内连接 左连接 右连接 全连接 0.准备1.内连接1.1 SQL(不带where)1.2 SQL&#xff08;带where&#xff09;1.3总结 2.左连接2.1SQL&#xff08;不带where&#xff09;2.2SQL&#xff08;带where&#xff09;2.3总结 3.右连接3.1 SQL&#xff08;不带where&#x…...

事务隔离:为什么你改了我还看不见

前提概要 你肯定不陌生&#xff0c;和数据库打交道的时候&#xff0c;我们总是会用到事务。最经典的例子就 是转账&#xff0c;你要给朋友小王转 100 块钱&#xff0c;而此时你的银行卡只有 100 块钱。 转账过程具体到程序里会有一系列的操作&#xff0c;比如查询余额、做加减法…...

吴恩达ChatGPT《LangChain Chat with Your Data》笔记

文章目录 1. Introduction2. Document Loading2.1 Retrieval Augmented Generation&#xff08;RAG&#xff09;2.2 Load PDFs2.3 Load YouTube2.4 Load URLs2.5 Load Notion 3. Document Splitting3.1 Splitter Flow3.2 Character Splitter3.3 Token Splitter3.4 Markdown Spl…...

https和http有什么区别

https和http有什么区别 简要 区别如下&#xff1a; ​ https的端口是443.而http的端口是80&#xff0c;且二者连接方式不同&#xff1b;http传输时明文&#xff0c;而https是用ssl进行加密的&#xff0c;https的安全性更高&#xff1b;https是需要申请证书的&#xff0c;而h…...

振弦采集仪及在线监测系统完整链条的岩土工程隧道安全监测

振弦采集仪及在线监测系统完整链条的岩土工程隧道安全监测 近年来&#xff0c;随着城市化的不断推进和基础设施建设的不断发展&#xff0c;隧道建设也日益成为城市交通发展的必需品。然而&#xff0c;隧道建设中存在着一定的安全隐患&#xff0c;如地质灾害、地下水涌流等&…...

linux基础学习

1.day1 2.day2 1、VIM配置&#xff1b; 2、安装SSH&#xff0c;调用putty接入终端&#xff1b; 3、shell命令&#xff1b; *&#xff1a;匹配任意长度的字符 &#xff1f;&#xff1a;匹配一个长度的字符 [...]&#xff1a;匹配其中指定的一个字符 [-]&#xff1a;匹配指定…...

android 前端常用布局文件升级总结(二)

问题一&#xff1a; android:name“android.support.v4.content.FileProvider” 报红 问题解决方案&#xff1a; 把xml布局文件里面: android.support.v4.content.FileProvider 更换成 androidx.core.content.FileProvider 问题二&#xff1a; android.support.design.wid…...

Linux复习——基础知识

作者简介:一名云计算网络运维人员、每天分享网络与运维的技术与干货。 座右铭:低头赶路,敬事如仪 个人主页:网络豆的主页​​​​​ 1. 有关早期linux系统中 sysvin的init的7个级别描述正确的是( )[选择1项] A. init 1 关机状态 B. init 2 字符界面多用户模式 …...

【数据结构】实验三:链表

实验三链表 一、实验目的与要求 1&#xff09;熟悉链表的类型定义&#xff1b; 2&#xff09;熟悉链表的基本操作&#xff1b; 3&#xff09;灵活应用链表解决具体应用问题。 二、实验内容 1&#xff09;请设计一个单链表的存储结构&#xff0c;并实现单链表中基本运算算…...

第4集丨webpack 江湖 —— loader的安装和使用

目录 一、loader简介1.1 使用 loader1.1.1 配置文件方式1.1.2 内联方式 1.2 loader 特性1.3 解析 loader1.4 命名规范 二、css loader的安装和使用2.1 安装2.2 配置2.3 测试 三、 less-loader 的安装和使用3.1 安装3.2 配置3.3 测试3.4 附件3.4.1 webpack.config.js3.4.2 index…...

【Lua学习笔记】Lua进阶——协程

文章目录 协程协程的定义和调度StatusRunning 协程 协程是一种并发操作&#xff0c;相比于线程&#xff0c;线程在执行时往往是并行的&#xff0c;并且线程在创建销毁执行时极其消耗资源&#xff0c;并且过长的执行时间会造成主进程阻塞。而协程可以以并发时轮值时间片来执行&…...

亚马逊云科技纽约峰会,充分释放数据价值和生成式AI的潜力

生成式AI将深刻改变每个公司的运营方式&#xff0c;标志着人工智能技术发展的新转折点。亚马逊云科技昨日在纽约峰会上宣布&#xff0c;推出七项生成式AI新功能&#xff0c;进一步降低了生成式AI的使用门槛&#xff0c;让无论是业务用户还是开发者都能从中受益。借助这些新功能…...