C语言动态内存管理(三)

news2024/10/10 22:22:08

目录

  • 五、C/C++程序的内存开辟
    • 1.图解
    • 2.关键点
  • 六、柔性数组
    • 1.什么是柔性数组
    • 2.两种语法形式
    • 3.柔性数组的特点
    • 4.柔性数组的创建及使用
      • 在这个方案中柔性数组的柔性怎么体现出来的?
    • 5.不用柔性数组,实现数组可大可小的思路
    • 6.对比
  • 总结

五、C/C++程序的内存开辟

1.图解

在这里插入图片描述
(1)内核空间:留给操作系统内核用,用户不能使用,自己写的程序没有权限访问内核空间
(2)栈区:存放局部变量
(3)内存映射段:(在Linux专栏中介绍)
(4)堆区:malloc,calloc,realloc动态开辟的空间

✨关于realloc在C语言动态内存管理(一)中讲了调整动态内存空间的作用,这里补充realloc函数的第一个参数是空指针NULL的时候,作用相当于开辟内存空间
举例:
realloc(NULL,20);
这句代码相当于malloc(20);

(5)数据段:存放全局变量,静态变量(静态变量不管是全局的还是局部的都放在数据段中)
(6)代码段:自己写的代码编译之后的汇编指令或者二进制指令

2.关键点

(1)栈区:
I.函数内的局部变量进作用域创建,出作用域销毁
II.向栈区申请的空间是有限的,不断地向栈区申请空间,就会出现栈溢出现象
III.栈区主要存放运行函数而分配的局部变量,函数参数,返回数据,返回地址等。
(2)堆区
一般由程序员分配释放。若程序员不释放,程序结束时可能由操作系统回收。分配方式类似于链表。
(3)数据段(静态区)
存放全局变量,静态数据。程序结束后由系统释放
(4)代码段
存放函数体(类成员函数和全局函数)的二进制代码

六、柔性数组

1.什么是柔性数组

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

2.两种语法形式

✨第一种:结构体中最后一个数组的大小指定为0

struct S {
	int n;
	int a[0];//柔性数组成员
};

✨第二种:结构体中的最后一个数组未指定大小

struct S {
	int n;
	int a[];//柔性数组成员
};

3.柔性数组的特点

(1)结构中的柔性数组成员前面必须至少有一个其他成员
(2)sizeof返回的这种结构大小不包含柔性数组的内存

也就是说包含柔性数组成员的结构体在用sizeof算这个结构体大小时,不计入柔性数组的大小

(3)包含柔性数组成员的结构用malloc函数进行内存的动态分配,并且分配的内存应该大于结构的大小,以适应柔性数组的预期大小
✨注意:
包含柔性数组成员的结构体创建时与一般结构体创建的方法不同。不能直接创建写成struct S ss;

4.柔性数组的创建及使用

柔性数组方案一

//柔性数组方案一
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<errno.h>

struct S {
	int n;
	char c;
	int arr[0];//柔性数组成员
};
int main() {
	struct S* ps = (struct S*)malloc(sizeof(struct S) + 10 * sizeof(int));
	/*
	(1)内存中放的是struct S类型的数据,开辟空间的地址应该是struct S*
	(2)结构体大小:包含除柔性数组成员的其他成员
	(3)10*sizeof(int)是柔性数组里打算放的东西,为柔性数组留了10个整型
	
	*/
	//判断是否开辟成功
	if (ps == NULL) {
		printf("%s\n",strerror(errno));
		return 1;
	}
	//使用
	ps->n = 55;
	ps->c = 'y';
	int i = 0;
	for (i = 0; i < 10; i++) {
		ps->arr[i] = i;
	}
	for(i = 0; i < 10; i++) {
		printf("%d\n", ps->arr[i]);
	}
	//调整arr数组的大小
	struct S* ptr = (struct S*)malloc(sizeof(struct S) + 20 * sizeof(int));
	if (ptr == NULL) {
		printf("%s\n", strerror(errno));
		return 1;
	}
	else {
		ps = ptr;//开辟成功的话最好用原来维护开辟好的空间的ps指针来维护这块空间
	}
	//使用
	//...

	//释放
	free(ps);
	ps = NULL;

	return 0;
}

在这里插入图片描述
在这里插入图片描述

在这个方案中柔性数组的柔性怎么体现出来的?

✨如果将结构体中arr长度写做10,结构体类型一旦创建好了,数组大小就不能改变了。不能每次使用的时候才进行更改数组的长度,因为一旦编译,结构体中的数组大小就固定了。
✨而柔性数组成员的空间是malloc出来的,malloc的空间可用realloc调整,那么这个数组的大小就柔性可变了。

5.不用柔性数组,实现数组可大可小的思路

结构中指针方案二

#include<stdio.h>
#include<stdlib.h>
struct S {
	int n;
	char c;
	int* arr;
};
int main() {
	struct S* ps = (struct S*)malloc(sizeof(struct S));
	if (ps == NULL) {
		perror("malloc");
		return 1;
	}
	int* ptr = (int*)malloc(10 * sizeof(int));
	if (ptr == NULL) {
		perror("malloc");
		return 1;
	}
	else {
		ps->arr = ptr;//开辟成功,把地址交给arr,arr指向这块空间
		/*
		arr指向这块空间,那么arr就可以像数组一样访问这块空间
		这块空间是malloc来的,所以不够时就可以用realloc进行扩容
		*/
	}

	//使用
	ps->n = 100;
	ps->c = 'y';
	int i = 0;
	for (i = 0; i < 10; i++) {
		ps->arr[i] = i;
		/*
		因为arr里面放的是这块空间的起始地址,和数组名一样
		把这块空间想象成连续的数组,数组名表示首元素的地址
		*/
	}
	//打印
	for (i = 0; i < 10; i++) {
		printf("%d ", ps->arr[i]);
	}
	//扩容-->调整arr数组的大小
	ptr = realloc(ps->arr, 20 * sizeof(int));
	if (ptr == NULL) {
		perror("realloc");
		return 1;
	}
	else {
		ps->arr = ptr;
	}

	//使用
	//...

	//释放
	//注意:释放动态内存开辟空间的顺序:后开辟的先释放
	free(ps->arr);
	ps->arr = NULL;
	free(ps);
	ps = NULL;

	return 0;
}

在这里插入图片描述
在这里插入图片描述

6.对比

柔性数组方案一
(1)malloc一次free一次
容易维护空间,不易出错
(2)malloc次数少,内存碎片较少,内存的使用率就会高一些
(3)有利于访问速度

柔性数组在内存空间中是连续的
涉及到局部性原理
如果内存是连续的,拿数据进行内存访问时,不是直接去拿,而是把内存中数据加载到寄存器里面。如果使用某个位置的数据,寄存器后面有空的话就会紧接着把它后面的数据加载到寄存器中。如果使用后面数据的话,在寄存器中已经拿到了,这样的效率更高。
在结构中指针方案二:
开辟了两块不连续的空间,在访问其中一块空间的时候,不断加载的事该空间后面的数据到寄存器,并没有把另一块空间的数据加载。访问另一块空间的数据时,寄存器中没有找到,跑到内存中去找,此时整体的效率就下降了。

总结

关于动态内存管理的全部知识就分享到这里啦,哪里有问题的话,欢迎各位朋友在评论区留言,可以点赞收藏博客关注后续的内容哦👻👻

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

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

相关文章

上手 SpringBoot

简介 SpringBoot设计的目的是简化 Spring应用的初始搭建以及 开发过程。 SpringBoot概述 parent 继承父pom文件&#xff0c;方便管理依赖的版本。此处涉及maven的使用 作用&#xff1a; 继承parent的形式可以采用引入依赖的形式实现效果 starter(原理是依赖传递) 包含了若…

C语言include学习

#include叫做文件包含命令&#xff0c;用来引入对应的头文件&#xff08;.h文件&#xff09;。 #include 也是C语言预处理命令的一种。 一个简单程序&#xff0c; #include <stdio.h>int main() {printf("hello world\n");return 0; } 使用printf()函数需要…

从虚拟现实到真实现实:沉浸式体验的未来 (ps beta)

首先&#xff0c;您需要创建一个免费的Adobe账户&#xff0c;如果您还没有一个账户的话。您可以在Adobe的注册页面上填写必要的信息来创建一个新账户。 Adobe: Creative, marketing and document management solutions login 创建账号&#xff1a; 编一个google邮箱&#xf…

F5 LTM 知识点和实验 1-BIG-IP 系统介绍

第一章&#xff1a;BIG-IP 系统介绍 基于包转发和全代理架构的区别&#xff08;packet-based vs full proxy architeture&#xff09; 一个网络设备如果是基于包转发的&#xff0c;则他是一个会话流&#xff08;communication stream&#xff09;的中间设备&#xff0c;而不是…

Atcoder Beginner Contest 311 C - E题讲解

C - Find it! 1. Description Problem Statement There is a directed graph with N N N vertices and N N N edges. The i i i-th edge goes from vertex i i i to vertex A i A_i Ai​. (The constraints guarantee that i ≠ A i i \neq A_i iAi​.) Find a dire…

记负均正 C语言实现

记负均正 描述 首先输入要输入的整数个数n&#xff0c;然后输入n个整数。输出为n个整数中负数的个数&#xff0c;和所有正整数的平均值&#xff0c;结果保留一位小数。 0即不是正整数&#xff0c;也不是负数&#xff0c;不计入计算。如果没有正数&#xff0c;则平均值为0。 数…

【C++】-多态的语法细节详解

&#x1f496;作者&#xff1a;小树苗渴望变成参天大树&#x1f388; &#x1f389;作者宣言&#xff1a;认真写好每一篇博客&#x1f4a4; &#x1f38a;作者gitee:gitee✨ &#x1f49e;作者专栏&#xff1a;C语言,数据结构初阶,Linux,C 动态规划算法&#x1f384; 如 果 你 …

软路由系统 --- iKuai虚拟机修改虚拟机网卡类型

软路由系统&#xff1a;iKuai 注意&#xff1a;2.0系列及以后版本&#xff0c;默认安装成功后只是绑定了LAN口&#xff0c;其他的接口需要自己进爱快的WEB界面手工绑定。 虚拟机网卡类型说明&#xff1a; 个人版vmware 建议选择Linux ubuntu64位&#xff0c;否则默认网卡类型…

MySQL 数据抽稀 每分钟取一条

假如原始数据为每5秒一个数据&#xff0c;现在想展示为每4分钟一条数据&#xff0c;先按照分钟数把除以4余数为0的行选出来&#xff0c;在按照 年月日 时分&#xff0c;做组内排序&#xff08;窗函数ROW_NUMBER&#xff09;&#xff0c;最后再拿出序号为1的行。 WITH data_01 …

基于SpringBoot+vue的私人健身与教练预约管理系统设计与实现(源码+LW+部署文档等)

博主介绍&#xff1a; 大家好&#xff0c;我是一名在Java圈混迹十余年的程序员&#xff0c;精通Java编程语言&#xff0c;同时也熟练掌握微信小程序、Python和Android等技术&#xff0c;能够为大家提供全方位的技术支持和交流。 我擅长在JavaWeb、SSH、SSM、SpringBoot等框架…

谈谈区块链技术

目录 1.什么是区块链 2.区块链的实现原理 3.区块链技术有哪些应用 4.区块链技术潜在的危害 5.区块链技术未来会怎么发展 1.什么是区块链 区块链是一种技术&#xff0c;它通过在一个分布式网络中记录和验证交易和数据&#xff0c;实现去中心化的数字账本。区块链的核心概念是…

读数据压缩入门笔记11_读后总结与感想兼导读

1. 基本信息 数据压缩入门 Understanding Compression [美]柯尔特麦克安利斯&#xff08;Colt McAnlis&#xff09;、[美]亚历克斯海奇 著&#xff1b; 王凌云 译 人民邮电出版社,2020年4月出版&#xff0c;1版 1.1. 读薄率 书籍总字数300千字&#xff0c;笔记总字数18197…

Linux用户权限

1.用户、权限、组的概念 1.1 用户 用户 是Linux系统工作中重要的一环&#xff0c;在Linux系统中&#xff0c;不论是由本机或是远程登录系统&#xff0c;每个系统都必须拥有一个账号&#xff0c;一个账号就是一个用户。 1.2 权限 在Linux系统中&#xff0c;每一个用户对不同…

汇编语言(第4版)实验5 编写、调试具有多个段的程序

&#xff08;1&#xff09;参考答案&#xff1a; ①不变&#xff08;0123h,0456h,0789h,0abch,0defh,0cabh,0987h&#xff09; ②076c 076b 076a ③X-2 X-1 &#xff08;2&#xff09;参考答案&#xff1a; ①不变&#xff08;0123h,0456h&#xff09; ②076c 076b 076a ③X-2…

AcWing 1275. 最大数—线段树单点修改

题目链接 AcWing 1275. 最大数 题目描述 分析 这道题是线段树的模板题&#xff0c;属于一眼ding真&#xff0c;鉴定为线段树的裸题&#xff0c;正好用来学习线段树。线段树、树状数组等数据结构题&#xff0c;难的地方是在如何分析出来要用哪个数据结构解决。 线段树要比树状…

上海VR全景展示,快速了解VR全景拍摄

导语&#xff1a; 随着科技的不断进步&#xff0c;虚拟现实技术的应用日益广泛。在这其中&#xff0c;VR全景图片作为一种数字化助力的全景拍摄方式&#xff0c;正逐渐成为人们关注的焦点。通过数字化技术&#xff0c;VR全景图片能够以360度全方位的视角呈现真实的场景&#x…

docker学习和进阶2023

文末有下载地址。如有侵权请联系作者谢谢&#xff01; docker学习和进阶2023 参考引用 docker学习笔记&#xff1a;https://blog.csdn.net/m0_46188681/article/details/128993319 Docker容器配置和资源限制&#xff1a;https://www.cnblogs.com/xiugeng/p/16254087.html 容…

idea使用命令将jar包导入到maven仓库中

因为今天突然忘了命令&#xff0c;记下来方便以后查看 pom文件的依赖 jar包路径 进入idea中命令窗 输入命令 mvn install:install-file -DfileD:\Project\spring-cloud\dubbo-api\target\dubbo-api-1.0-SNAPSHOT.jar -DgroupIdcom.wmx -DartifactIddubbo-api -Dversion1.0…

flutter数字动画库:animated_flip_counter

前言 在数字动画的制作中&#xff0c;有时候我们会面临时间紧张或效果不尽如人意的情况。这时&#xff0c;使用现成的动画库或工具可以大大提高效率&#xff0c;同时也能够获得更好的效果。animated_flip_counter就是一个非常不错的数字动画库&#xff0c;可以帮助我们快速地创…

Leetcode-每日一题【剑指 Offer II 010. 和为 k 的子数组】

题目 给定一个整数数组和一个整数 k &#xff0c;请找到该数组中和为 k 的连续子数组的个数。 示例 1&#xff1a; 输入:nums [1,1,1], k 2输出: 2解释: 此题 [1,1] 与 [1,1] 为两种不同的情况 示例 2&#xff1a; 输入:nums [1,2,3], k 3输出: 2 提示: 1 < nums.leng…