C语言-动态内存申请

一、动态分配内存的概述

在数组一章中,介绍过数组的长度是预先定义好的,在整个程序中固定不变,但是在实际的编程中,往往会发生这种情况,即所需的内存空间取决于实际输入的数据,而无法预先确定 。为了解决上述问题,C语言提供了一些内存管理函数,这些内存管理函数可以按需要动态的分配内存空间,也可把不再使用的空间回收再次利用。

动态分配内存就是在堆区开辟空间

二、静态分配、动态分配

静态分配

1、 在程序编译或运行过程中,按事先规定大小分配内存空间的分配方式。int a [10]

2、 必须事先知道所需空间的大小。

3、 分配在栈区或全局变量区,一般以数组的形式。

4、 按计划分配。

动态分配

1、在程序运行过程中,根据需要大小自由分配所需空间。

2、按需分配。

3、分配在堆区,一般使用特定的函数进行分配。

三、动态分配函数

3.1 malloc

1 #include <stdlib.h>

2 void *malloc(unsigned int size);

3 功能:在堆区开辟指定长度的空间,并且空间是连续的

4 参数:

5 size:要开辟的空间的大小

6 返回值:

7 成功:开辟好的空间的首地址

8 失败:NULL

注意

1、在调用malloc之后,一定要判断一下,是否申请内存成功。

2、如果多次malloc申请的内存,第1次和第2次申请的内存不一定是连续的

3、使用malloc开辟空间需要保存开辟好的空间的首地址,但是由于不确定空间用于做什么,所以本身返回值类型为void *,所以在调用函数时根据接收者的类型对其进行强制类型转换

1 #include <stdio.h>

2 #include <stdlib.h>

3

4 char *fun()

5 {

6 //char ch[100] = "hello world";

7

8 //静态全局区的空间只要开辟好,除非程序结束,否则不会释放,所以

9 //如果是临时使用,不建议使用静态全局区的空间

10 //static char ch[100] = "hello world";

11

12 //堆区开辟空间,手动申请手动释放,更加灵活

13 //使用malloc函数的时候一般要进行强转

14 char *str = (char *)malloc(100 * sizeof(char));

15 str[0] = 'h';

16 str[1] = 'e';

17 str[2] = 'l';

18 str[3] = 'l';

19 str[4] = 'o';

20 str[5] = '\0';

21

22 return str;

23 }

24

25 int main(int argc, char *argv[])

26 {

27 char *p;

28 p = fun();

29 printf("p = %s\n", p);

30

31 return 0;

32 }

3.2 free

1 #include <stdlib.h>2 void free(void *ptr)

3 功能:释放堆区的空间

4 参数:

5 ptr:开辟后使用完毕的堆区的空间的首地址

6 返回值:

7

注意:

free函数只能释放堆区的空间,其他区域的空间无法使用free

free释放空间必须释放malloc或者calloc或者realloc的返回值对应的空间,不能说只释放一部分。

free(p); 注意当free后,因为没有给p赋值,所以p还是指向原先动态申请的内存。但是内存已经不能再用了,p变成野指针了,所以一般为了放置野指针,会free完毕之后对p赋为NULL。

一块动态申请的内存只能free一次,不能多次free 。

3.3 calloc

1 #include <stdlib.h>

2 void * calloc(size_t nmemb,size_t size);

3 功能:在堆区申请指定大小的空间

4 参数:

5 nmemb:要申请的空间的块数

6 size:每块的字节数

7 返回值:

8 成功:申请空间的首地址

9 失败:NULL

注意:

malloc和calloc函数都是用来申请内存的。

区别:

1) 函数的名字不一样

2) 参数的个数不一样

3) malloc申请的内存,内存中存放的内容是随机的,不确定的,

而calloc函数申请的内存中的内容为0

例如:

char *p=(char *)calloc(3,100);在堆中申请了3块,每块大小为100个字节,即300个字节连续的区域。

3.4 realloc

1 #include <stdlib.h>

2 void* realloc(void *s,unsigned int newsize);

3 功能:在原本申请好的堆区空间的基础上重新申请内存,新的空间大小为函数的第二个参数

4 如果原本申请好的空间的后面不足以增加指定的大小,系统会重新找一个足够大的位

5 置开辟指定的空间,然后将原本空间中的数据拷贝过来,然后释放原本的空间

6 如果newsize比原先的内存小,则会释放原先内存的后面的存储空间,

7 只留前面的newsize个字节

8 参数:

9 s:原本开辟好的空间的首地址

10 newsize:重新开辟的空间的大小

11 返回值:

12 新的空间的首地址

增加空间:

1 char *p;

2 p=(char *)malloc(100)

3 //想在100个字节后面追加50个字节

4 p=(char *)realloc(p,150);//p指向的内存的新的大小为150个字节

减少空间:

1 char *p;

2 p=(char *)malloc(100)

3 //想重新申请内存,新的大小为50个字节4 p=(char *)realloc(p,50);//p指向的内存的新的大小为50个字节,100个字节的后50个字节的存储空间就被释放了

注意:malloc calloc relloc 动态申请的内存,只有在free或程序结束的时候才释放。

四、内存泄漏

内存泄露的概念:

申请的内存,首地址丢了,找不了,再也没法使用了,也没法释放了,这块内存就被泄露了。

内存泄漏案例1:

1 int main()

2 {

3 char *p;

4 p=(char *)malloc(100);

5 //接下来,可以用p指向的内存了

6

7 p="hello world";//p指向别的地方了,保存字符串常量的首地址

8

9 //从此以后,再也找不到你申请的100个字节了。则动态申请的100个字节就被泄露了

10

11 return 0;

12 }

内存泄漏案例2:

1 void fun()

2 {

3 char *p;

4 p=(char *)malloc(100);

5 //接下来,可以用p指向的内存了

6 ...

7 }

8

9 int main()

10 {

11 //每调用一次fun泄露100个字节

12 fun();

13 fun();14 return 0;

15 }

解决方式1:

1 void fun()

2 {

3 char *p;

4 p=(char *)malloc(100);

5 //接下来,可以用p指向的内存了

6 ...

7 free(p);

8 }

9

10 int main()

11 {

12 fun();

13 fun();

14 return 0;

15 }

解决方式2:

1 char * fun()

2 {

3 char *p;

4 p=(char *)malloc(100);

5 //接下来,可以用p指向的内存了

6 ...

7 return p;

8 }

9

10 int main()

11 {

12 char *q;

13 q=fun();

14 //可以通过q使用 ,动态申请的100个字节的内存了

15 //记得释放

16 free(q);

17 //防止野指针

18 q = NULL;19

20 return 0;

21 }

总结:申请的内存,一定不要把首地址给丢了,在不用的时候一定要释放内存。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

一、创建Vue3项目

1. 下载 node.js 下载地址&#xff1a;https://nodejs.org/zh-cn 优先选择 16 版本; node -v || node -version 可以检查本地 node.js 版本 2. 设置淘宝镜像源 npm config set registry https://registry.npmmirror.com/ 设置淘宝镜像源 npm config get registry 查看当前镜像…

常见的网络安全威胁和防护方法

随着数字化转型和新兴技术在各行业广泛应用&#xff0c;网络安全威胁对现代企业的业务运营和生产活动也产生了日益深远的影响。常见的网络安全威胁通常有以下几种&#xff1a; 1. 钓鱼攻击 攻击者伪装成合法的实体&#xff08;如银行、电子邮件提供商、社交媒体平台等&#xf…

解析PDF二维码:数字时代文件管理的创新之道

随着数字时代的来临&#xff0c;文件管理方式正经历着翻天覆地的变革。在这个变革的浪潮中&#xff0c;PDF二维码作为一种创新的技术手段&#xff0c;正逐渐引起人们的关注。本文将深入探讨PDF二维码的概念、应用领域以及在文件管理中的前景。 一、PDF二维码的概念 PDF二维码…

mcu专用看门狗复位芯片(如MAX706)

mcu专用看门狗复位芯片&#xff08;如MAX706&#xff09; 为什么要使用电压复位芯片RESET引脚WDO引脚MR引脚WDI引脚 国产替代型号应用电路1 推荐电路&#xff08;用一个跳线帽使能/关闭看门狗功能&#xff0c;调试MCU时防止看门狗芯片随便触发复位功能&#xff09;&#xff0c;…

Linux操作系统概述

操作系统&#xff08;Operating System&#xff09;的定义 操作系统&#xff0c;是指直接管理系统硬件和资源&#xff08;如 CPU、内存和存储空间&#xff09;的软件。 操作系统的基本功能 ①统一管理计算机资源&#xff1a;处理器资源&#xff0c;IO设备资源&#xff0c;存储…

Flutter的安装与环境配置

一、下载安装Futter&#xff1a; 1、Flutter中文文档&#xff1a; 安装和环境配置 - Flutter 中文文档 - Flutter 中文开发者网站 - Flutter 2、下载 Futter SDK&#xff1a; Flutter中文文档 里面有&#xff0c;下载完成之后找个文件夹解压出来&#xff0c;最好不要将 Flu…

MybatisPlus应用参数类型不一致导致索引失效

业务场景介绍 在电商项目中&#xff0c;有一个商品表【t_goods】和一个商品sku表【t_goods_sku】,具体表结构如下所示&#xff1a; CREATE TABLE t_goods (id bigint NOT NULL AUTO_INCREMENT COMMENT 主键id,brand_id varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_…

Java链表(2)

&#x1f435;本篇文章将对双向链表进行讲解&#xff0c;模拟实现双向链表的常用方法 一、什么是双向链表 双向链表在指针域上相较于单链表&#xff0c;每一个节点多了一个指向前驱节点的引用prev以及多了指向最后一个节点的引用last&#xff1a; 二、双向链表的模拟实现 首先…

Python实现时间序列分析AR定阶自回归模型(ar_select_order算法)项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档视频讲解&#xff09;&#xff0c;如需数据代码文档视频讲解可以直接到文章最后获取。 1.项目背景 时间序列分析中&#xff0c;AR定阶自回归模型&#xff08;AR order selection&#xff09;是指确定自回…

GPT栏目:yarn 安装

GPT栏目&#xff1a;yarn 安装 一、前言 在跟GPT交互的时候&#xff0c;发现最近gpt4给出的答案率有了比较明显的提高&#xff0c;简单记录一下&#xff0c;我用gpt4拿到的答案吧。 本人已按照这个步骤成功 二、具体步骤 要安装 yarn&#xff0c;你可以按照以下步骤进行操作…

Ubuntu 22.04.1 LTS 编译安装 nginx-1.22.1,Nginx动静分离、压缩、缓存、黑白名单、跨域、高可用、性能优化

1.Ubuntu 22.04.1 LTS 编译安装 nginx-1.22.1 1.1安装依赖 sudo apt install libgd-dev 1.2下载nginx wget http://nginx.org/download/nginx-1.22.1.tar.gz 1.3解压nginx tar -zvxf nginx-1.22.1.tar.gz 1.4编译安装 cd nginx-1.22.1 编译并指定安装位置&#xff0c;执行安装…

Vue学习笔记之生命周期函数

生命周期示意图如下所示&#xff1a; beforeCreate&#xff1a;组件初始化之前触发该事件created&#xff1a;组件初始化完毕触发该事件beforeMount&#xff1a;Vue应用对象挂载DOM结点之前触发该事件mounted&#xff1a;DOM结点挂载成功之后触发该事件beforeUpdate&#xff1a…

欧拉计划第816题:求大量点的最短距离

本次来解决欧拉计划的第816题: 解: 第一步:最原始的算法 先从简单的情况开始,即原题里的14个点的情况 import mathdef gen_points(n):s = [0] * (2*n)s[0] = 290797for i in range(1, 2*n):s[i] = (s[i - 1] * s[i - 1]) % 50515093p = [(s[2 * i], s[2 * i + 1]) for…

分布式ID是什么,以美团Leaf为例改造融入自己项目【第十一期】

前言 在日常开发中&#xff0c;主键id应用是非常广泛的&#xff0c;但是当涉及到分布式系统的时候&#xff0c;往往需要使用到分布式id&#xff0c;每一个服务里面一套生成规则的不易管理&#xff0c;容易引发冲突。我的IM聊天系统中使用分布式id来生成消息唯一键,为后面幂等做…

OpenHarmony RK3568 启动流程优化

目前rk3568的开机时间有21s&#xff0c;统计的是关机后从按下 power 按键到显示锁屏的时间&#xff0c;当对openharmony的系统进行了裁剪子系统&#xff0c;系统app&#xff0c;禁用部分服务后发现开机时间仅仅提高到了20.94s 优化微乎其微。在对init进程的log进行分析并解决其…

12V-80V车灯芯片都有哪些?-H5028L

电动车车灯芯片的工作原理可以简要概括为以下几点&#xff1a; 光源&#xff1a;电动车车灯通常使用LED&#xff08;Light Emitting Diode&#xff09;作为光源。LED是一种半导体器件&#xff0c;当电流通过LED时&#xff0c;它会发光。 驱动电路&#xff1a;车灯芯片中包含驱…

百度智能小程序开发平台:SEO关键词推广优化 带完整的搭建教程

移动互联网的普及&#xff0c;小程序成为了众多企业和开发者关注的焦点。百度智能小程序开发平台为开发者提供了一站式的解决方案&#xff0c;帮助企业快速搭建并推广自己的小程序。本文将重点介绍百度智能小程序开发平台的SEO关键词推广优化功能&#xff0c;并带完整的搭建教程…

保护医疗数据不受威胁:MPLS专线在医疗网络安全中的角色

随着数字技术的快速发展&#xff0c;医疗行业正在经历一场革命。从电子健康记录到远程医疗服务&#xff0c;数字化不仅提高了效率&#xff0c;也带来了前所未有的挑战--尤其是关于数据安全和隐私保护的挑战。在这样的背景下&#xff0c;如何确保敏感的医疗数据安全传输&#xf…

github添加 SSH 密钥

1 打开终端 输入 ssh-keygen -t rsa -b 4096 -C "github邮箱地址"如果不需要密码可以一路回车 出现这个页面就是生存成功了 open ~/.ssh // 打开.ssh 找到id_rsa.pub复制出内容新建ssh密钥输入内容,保存即可

本地部署Tale博客网站并结合内网穿透实现公网访问本地站点

文章目录 前言1. Tale网站搭建1.1 检查本地环境1.2 部署Tale个人博客系统1.3 启动Tale服务1.4 访问博客地址 2. Linux安装Cpolar内网穿透3. 创建Tale博客公网地址4. 使用公网地址访问Tale 前言 今天给大家带来一款基于 Java 语言的轻量级博客开源项目——Tale&#xff0c;Tale…