C语言柔性数组详解与应用

目录

介绍什么是柔性数组

柔性数组有什么特点

柔性数组使用场景

模拟实现柔性数组

柔性数组的好处


介绍什么是柔性数组

柔性数组是C语言引入的一个新特性,在C99中我们把最后一个没有指明大小的数组叫做是柔性数组。

我们这里举一个栗子:

cpp 复制代码
struct test {
    int size;
    int data[]; // 柔性数组成员
};

除此之外我们还有一个非C99标准定义的柔性数组:

cpp 复制代码
struct test {
    int size;
    int data[0]; // 柔性数组
}

柔性数组有什么特点

1、大小不固定

柔性数组的大小在编译的阶段是无法确定的,需要到了运行的时候才动态地分配空间。

我们在使用sizeof的时候是不包括柔性数组的:

cpp 复制代码
#include <stdio.h>

struct test {
    int size;
    int data[];
};
int main() {
    printf("%d\n", sizeof(struct test)); // 这里的结果是4
    return 0;
}

2、出现在结构体的最后一个成员且至少有一个其他成员

如果不是有一个其他成员,代码会报错:

cpp 复制代码
#include <stdio.h>

struct test {
    int data[];
};

3、如果我们想使用柔性数组来存储数据,那么我们就需要开辟比原来结构体大的空间。

cpp 复制代码
#include <stdio.h>

struct test {
    int size;
    int data[];
};
int main() {
    struct test* p = (struct test*)malloc(sizeof(struct test) + 5 * sizeof(int));
    return 0;
}

柔性数组使用场景

我们通常是在要改变数组大小的情况下使用柔性数组的。

我们这里举一个例子,就是要求结构体的数组从存储0到4变成存储5到9:

cpp 复制代码
#include <stdio.h>
#include <stdlib.h>
struct test {
    int size;
    int data[];
};
int main() {
    struct test* p = (struct test*)malloc(sizeof(struct test) + 5 * sizeof(int));
    p->size = 10;
    for(int i = 0; i < 5; i++) {
        p->data[i] = i;
    } 
    struct test* tp = (struct test*)realloc(p, sizeof(struct test) + 10 * sizeof(int));
    if(tp != NULL) {
        p = tp;
    }
    for(int i = 5; i < 10; i++) {
        p->data[i] = i;
    }
    // 别忘了要释放开辟的空间^_^
    free(p);
    p = NULL;
    return 0;
}

模拟实现柔性数组

事实上,如果我们有了之前内存管理相关的知识储备,我们就可以自己实现上面的功能:

cpp 复制代码
#include <stdio.h>
#include <stdlib.h>

struct test {
    int size;
    int* data;
};
int main() {
    struct test* p = (struct s*)malloc(sizeof(struct test));
    p->data = (int*)malloc(5 * sizeof(int));
    p->size = 10;
    for(int i = 0; i < 5; i++) {
        p->data[i] = i;
    }
    int* tp = (int*)realloc(p->data, 10 * sizeof(int));
    if(tp != NULL) {
        p->data = tp;
    }
    for(int i = 5; i < 10; i++) {
        p->data[i] = i;
    }
    free(p->data);
    p->data = NULL;
    free(p);
    p = NULL;
    return 0;
}

其实柔性数组就是一种C语言提供灵活内存管理的特性,它允许我们在结果体内部定义一个大小不确定的数组,在运行的时候使用动态内存分配为结构体和数组分配内存,实际上它是为了给我们处理动态大小变化的数据提供一种便利的方式。

柔性数组的好处

1、方便释放内存

我们在上面模拟实现的时候也发现了,我们使用柔性数组只要释放一次内存,模拟的时候就需要两次而且还要注意到释放顺序的问题。

2、提高访问的速度

我们使用柔性数组的时候,内存中开辟的空间是连续的:

我们自己实现的就是下面这种情况了:

我们这里可以发现第一种情况的内存是连续的,而第二种就不是了,我们可以打印出来看看就知道了。

其实我们在考虑------连续和不连续又有什么影响呢?其实这里就涉及到了一个重要的概念,就是内存碎片

越多的不连续的内存分配就会产生越多的内存碎片,这样就会使得我们的内存利用率低下。

我们这里还有一个概念,就是CPU在存储器取数据的时候,会遵循局部性原理(计算机组成原理):也就是CPU在访问存储器的时候,无论是取数据还是指令,访问的内存单元往往都是聚集在了一个比较小的连续的空间中。所以,将相关联的数据存储在一起是会提高CPU的访问速度的。

相关推荐
吾疾唯君医几秒前
记录GoLang创建文件并写入文件的中文乱码错误!
开发语言·后端·golang
小年糕是糕手20 分钟前
【数据结构】算法复杂度
c语言·开发语言·数据结构·学习·算法·leetcode·排序算法
JAVA学习通28 分钟前
微服务项目->在线oj系统(Java-Spring)--C端用户(超详细)
java·开发语言·spring
数据知道28 分钟前
Go基础:Go语言ORM框架GORM详解
开发语言·jvm·后端·golang·go语言
计算机毕业设计小帅28 分钟前
【2026计算机毕业设计】基于jsp的毕业论文管理系统
java·开发语言·毕业设计·课程设计
明天会有多晴朗30 分钟前
深度剖析 C++ 之内存管理篇
c语言·开发语言·c++
potato_may1 小时前
C语言第3讲:分支和循环(上)—— 程序的“决策”与“重复”之旅
c语言·开发语言
kalvin_y_liu1 小时前
【MES架构师与C#高级工程师(设备控制方向)两大职业路径的技术】
开发语言·职场和发展·c#·mes
xxxxxxllllllshi1 小时前
Java 代理模式深度解析:从静态到动态,从原理到实战
java·开发语言·笔记·算法·代理模式
计算机毕业设计指导1 小时前
从零开始构建HIDS主机入侵检测系统:Python Flask全栈开发实战
开发语言·python·flask