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

柔性数组与c/c++程序中内存区域的划分

1.柔性数组

1.1柔性数组的定义

柔性数组是指在结构体中定义的,其大小在编译时未确定,而在运行时动态分配的数组。这种数组允许结构体的大小根据需要动态变化。语法如下:

struct D
{int a;int arry1[0];
};struct F
{int a;int arry2[];
};

柔性数组有上述两种写法,具体实现依据编译器,vs编译器都支持

1.2柔性数组的特点

(1)结构体中,在柔性数组成员之前至少还有一个成员

(2)柔性数组属于结构体的最后一个成员

(3)sizeof计算的结构体大小不包括柔性数组的大小

(4)包含柔性数组的结构体使用malloc分配的内存需要大于结构体的大小,以适应柔性数组的预期大小

1.3柔性数组的使用

#include <stdio.h>
#include <stdlib.h>struct D
{int a;int arry1[0];
};int main()
{struct D * p = (struct D* )malloc(sizeof(struct D) + 10 * sizeof(int));if (p == NULL){perror("malloc");return 1;}for (int i = 0; i < 10; ++i)p->arry1[i] = i;struct D * temp = (struct D*)realloc(p, sizeof(struct D) + 20* sizeof(int));if (temp != NULL)p = temp;for (int i = 0; i < 20; ++i)printf("%d ", p->arry1[i]);free(p);p = NULL;return 0;
}

(1)柔性数组之所以有柔性是因为该数组的大小可以根据需要去改变

即使是变长数组,程序运行之后,大小也会被固定

(2)上述的结构体成员存在于堆区上,也可以通过下面的方式进行实现

在堆区创建一个包含指针的结构体

在堆区分配一定大小的内存,用结构体的指针指向这块内存

#include <stdio.h>
#include <stdlib.h>struct D
{int a;int *arry1;
};int main()
{struct D* p = (struct D*)malloc(sizeof(struct D));if (p == NULL){perror("malloc");return 1;}int* temp = (int*)malloc(20 * sizeof(int));if (temp != NULL)p->arry1 = temp;for (int i = 0; i < 20; ++i)p->arry1[i] = i;for (int i = 0; i < 20; ++i)printf("%d ", p->arry1[i]);free(p->arry1);p-> arry1 = NULL:free(p);p = NULL;return 0;
}

1.3.2两种方法的总结

(1)第一种方法是 

在堆区开辟一块连续的内存,使用一次malloc和free即可

struct D
{int a;int arry[0];
};

 (2)第二种方法不一定连续,需要使用两次malloc和free


struct D
{int a;int *arry;
};

 

相比之下,方法1具有两个好处:

(1)方便内存释放:一次free即可

(2)有利于访问速度:连续的内存访问速度更快

2.c/c++程序中内存区域的划分

主要针对,学习程序语言时的划分

图片来源:鹏哥c语言

(1)栈区(stack):主要存放运⾏函数⽽分配局部变量、函数参数、返回数据、返回地址

特点:

1.内存分配和释放速度快

2.大小有限,较大的局部变量或深度递归可能导致栈溢出

3.在执⾏函数时,函数内局部变量的存储单元都可以在栈上创建,函数执⾏结束时 这些存储单元⾃动被释放

(2)堆区(heap):用于动态内存分配

特点:

1.由程序员手动分配和释放(使用mallocnew等函数,以及freedelete等函数)

2.如果程序员不释放内存,则可能会导致内存泄漏

3.堆区的大小受限于计算机系统中有效的虚拟内存,相对灵活且较大

(3)数据段(静态区):用于存储全局变量和静态变量。

特点:

1.数据段在程序编译时已经分配了内存并初始化了初始值

2.未初始化的全局变量和静态变量在程序开始执行时会被自动初始化为零对于数值类型变量

3.通常包括全局变量、静态变量以及字符串常量(尽管字符串常量本身可能存储在只读区域,但其引用或指针可能存储在数据段中)

(4)代码段用于存储程序的机器指令

http://www.lryc.cn/news/531873.html

相关文章:

  • mini-lsm通关笔记Week2Day7
  • Typora免费使用
  • AI驱动的无线定位:基础、标准、最新进展与挑战
  • 苹果再度砍掉AR眼镜项目?AR真的是伪风口吗?
  • 18 大量数据的异步查询方案
  • DRM系列八:Drm之DRM_IOCTL_MODE_ADDFB2
  • 软件测试用例篇
  • PopupMenuButton组件的功能和用法
  • Python进行模型优化与调参
  • vue2-组件通信
  • 20250205确认荣品RK3566开发板在Android13下可以使用命令行reboot -p关机
  • 设计模式---观察者模式
  • 初八开工!开启数字化转型新征程!
  • 文本分析NLP的常用工具和特点
  • DeepSeek 与 ChatGPT 对比分析
  • vite---依赖优化选项esbuildOptions详解
  • ElasticSearch 学习课程入门(二)
  • 使用 Redis Streams 实现高性能消息队列
  • 深度学习|表示学习|卷积神经网络|DeconvNet是什么?|18
  • (优先级队列(堆)) 【本节目标】 1. 掌握堆的概念及实现 2. 掌握 PriorityQueue 的使用
  • 优化数据库结构
  • 密云生活的初体验
  • 图像分类与目标检测算法
  • 计算机网络——流量控制
  • 体验 DeepSeek 多模态大模型 Janus-Pro-7B
  • 使用mockttp库模拟HTTP服务器和客户端进行单元测试
  • 解决每次打开终端都需要source ~/.bashrc的问题(记录)
  • UE5 蓝图学习计划 - Day 14:搭建基础游戏场景
  • C++常用拷贝和替换算法
  • 取消和确认按钮没有显示的问题