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

【C语言】柔性数组

柔性数组

  • 1. 柔性数组介绍
  • 2. 柔性数组特点
  • 3. 用例
    • 3.1 代码一:
    • 3.2 代码二:
  • 4. 柔性数组优势:

1. 柔性数组介绍

也许你从来没有听说过柔性数组(flexible array)这个概念,但是它确实是存在的。

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

例如:零长度数组概念

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

有些编译器无法编译,可以改成:

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

2. 柔性数组特点

  • 结构中的柔性数组成员前面必须至少一个其他成员。

  • sizeof 返回的这种结构大小不包括柔性数组的内存。

  • 包含柔性数组成员的结构用malloc ()函数进行内存的动态分配,并且分配的内存应该大于结构的大小,
    以适应柔性数组的预期大小。

3. 用例

3.1 代码一:

在这里插入图片描述

//代码一:
struct S
{
	int n;
	char c;
	int arr[];
};
int main()
{
	struct S* ps = (struct S*)malloc(sizeof(struct S) + 10 * sizeof(int));
                                         //设置int arr数组元素为10个,10*sizeof(int)
	if (ps == NULL)
	{
		printf("%s\n", strerror(errno));
		return 1;
	}
	ps->n = 100;
	ps->c = 'w';
	for (int i = 0;i < 10;i++)
	{
		ps->arr[i] = i;
	}
	for (int i = 0;i < 10;i++)
	{
		printf("%d\n", ps->arr[i]);
	}
	free(ps);
	ps = NULL;
}

3.2 代码二:

在这里插入图片描述

//代码二:
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));
                     //设置int arr数组元素为10个,10*sizeof(int)
 if (ptr == NULL)
 {
  perror("malloc2");
  return 1;
 }
 else
 {
  ps->arr = ptr;
 }
 //使用
 ps->n = 100;
 ps->c = 'w';
 int i = 0;
 for (i = 0; i < 10; i++)
 {
  ps->arr[i] = i;
 }
 //打印
 for (i = 0; i < 10; i++)
 {
  printf("%d ", ps->arr[i]);
 }


 //扩容 - 调整arr的大小
 ptr = realloc(ps->arr, 20 * sizeof(int));
                     //增容int arr数组元素为20个,10*sizeof(int)
 if (ptr == NULL)
 {
  perror("realloc");
  return 1;
 }
 else
 {
	 ps->arr = ptr;
 }


 //使用
 ps->n = 100;
 ps->c = 'w';
 for (i = 0; i < 20; i++)
 {
	 ps->arr[i] = i;
 }
 //打印
 for (i = 0; i < 20; i++)
 {
	 printf("%d ", ps->arr[i]);
 }


 //释放
 free(ps->arr);
 ps->arr = NULL;
 free(ps);
 ps = NULL;

 return 0;
}

4. 柔性数组优势:

代码一和代码二都可以完成相同功能,但代码一有以下两个好处:

第一个好处是:方便内存释放。

如果我们的代码是在一个给别人用的函数中,你在里面做了二次内存分配,并把整个结构体返回给用户。

用户调用free可以释放结构体,但是用户并不知道这个结构体内的成员也需要free,所以你不能指望用户来发现这个事。

所以,如果我们把结构体的内存以及其成员要的内存一次性分配好了,并返回给用户一个结构体指针,用户做一次free就可以把所有的内存也给释放掉。
在这里插入图片描述

第二个好处是:这样有利于访问速度。

连续的内存有益于提高访问速度,也有益于减少内存碎片。

(其实,我个人觉得也没多高了,反正 你跑不了要用做偏移量的加法来寻址)

补充:局部性原理:

  • 局部性原理是计算机科学中的一种基本原理,指的是程序中的指令和数据在执行时会集中在较小的一部分存储器中,并且在短时间内重复使用。这种局部性表现在两个方面:时间局部性和空间局部性。

  • 时间局部性指的是,某个数据或指令一旦被使用,在短时间内会被多次使用。例如,在循环结构的程序中,循环体内的指令会被反复执行。

  • 空间局部性指的是,某个数据或指令的使用会对其周围的数据与指令产生影响,这些数据和指令也会在短时间内被多次使用。例如,在数组操作中,相邻的元素都会被频繁地访问。

  • 局部性原理的应用可以提高计算机系统的性能,例如通过缓存技术和虚拟内存技术,让系统可以快速地访问常用的数据和代码,从而减少了额外的访问时间和存储开销。

👊👊👊
感谢阅读!


http://www.kler.cn/a/3544.html

相关文章:

  • 计算机组成原理(计算机系统3)--实验八:处理器结构拓展实验
  • ARM-V9 CCA/RME QEMU环境搭建
  • 客户案例:电商平台对帐-账单管理(亚马逊amazon)
  • Spring WebFlux 和 Spring MVC 的主要区别是什么?
  • sql实战解析-sum()over(partition by xx order by xx)
  • 论文笔记(六十三)Understanding Diffusion Models: A Unified Perspective(一)
  • 【数据结构】顺序表的深度刨剖析
  • 批量保存网页为单个网页文件
  • 【差分数组】
  • Vue学习计划九:了解Vue动画效果以及过渡动画和动态组件的使用方法
  • Spring容器实现原理-Spring的结构组成与核心类
  • [golang gin框架] 6.Gin GORM简介以及安装
  • 基于51单片机的室内湿度加湿温度声光报警智能自动控制装置设计
  • 数字图像处理 纹理分析方法简略综述
  • 快速求解组合数
  • 【微信小程序】-- 页面导航 -- 编程式导航(二十三)
  • chartgpt 告诉我的,loss 函数的各种知识
  • Android DataBinding 自定义View实现数据双向绑定
  • 中国蚁剑AntSword实战
  • 【Linux】Linux下权限的理解
  • 大公司为什么禁止SpringBoot项目用Tomcat?
  • 学习typeScript(weakMap,weakSet,set,map)
  • 动态规划---线性dp和区间dp
  • STM32外设-定时器详解
  • QT之QSysInfo(查看电脑信息)
  • 【springcloud 微服务】Spring Cloud Alibaba Nacos使用详解