【C语言进阶】柔性数组

学习了动态内存管理和结构体之后,我们可以开始学习柔性数组~

目录

[1. 什么是柔性数组](#1. 什么是柔性数组)

2.柔性数组的特点

[2.1 柔性数组两种开辟空间的方式](#2.1 柔性数组两种开辟空间的方式)


1. 什么是柔性数组

在结构体中的最后一个成员是一个没有指定大小的数组,那么这个数组叫做柔性数组。

cpp 复制代码
typedef struct st_type
{
    int i;
    int a[0]; // 柔性数组成员
};

typedef struct st_type
{
    int i;
    int a[]; // 编译不通过可以将0去掉
};

2.柔性数组的特点

柔性数组具有以下特点:

①结构体中的柔性数组之前必须至少有一个其他成员。

②sizeof返回的结构体大小不包含柔性数组的内存。

③包含柔性数组的结构体用malloc函数进行内存的动态分配,分配的内存应该大于结构体的大小,以适应柔性数组的预期大小。

最后一点是什么意思呢,我们可以看下面的图片,我们使用malloc创建结构体变量,除了结构体本身的4个字节,我们还需要给柔性数组分配40个字节的变量,这里将返回的地址给了ps,内存布局如下图所示:

2.1 柔性数组两种开辟空间的方式

由于arr数组是由malloc分配的堆空间,所以这里可以使用relloc进行扩容。,如下图所示,我们又增加了80个字节,当然开辟完毕之后需要判断是否为空指针。

cpp 复制代码
struct S 
{
    int n;
    int* arr;
};



int main()
{
    
    struct S* ps = (struct S*)malloc(sizeof(struct S) + 40);
    ps -> n = 100;
    for(int i = 0;i < 10; i++)
    {
        ps->arr[i] = i;
    }
    struct S* ptr = (struct S*)realloc(ps,sizeof(struct S) + 80);
    if(ptr != NULL)
    {
        ps = ptr;
    }

    return 0;
}

当使用完毕之后,我们只需要进行手动释放即可。

除了想上面那样直接一次开辟空间,我们还可以多次开辟空间达到效果;

cpp 复制代码
struct S 
{
    int n;
    int* arr;
};



int main()
{
    
    struct S* ps = (struct S*)malloc(sizeof(struct S));
    if(ps == NULL)
    {
        return 1;
    }
    ps -> n= 100;
    ps ->arr =  (int*)malloc(40);
    if(ps -> arr == NULL)
    {
        return 1;
    }
    for(int i  = 0;i < 10;i++)
    {
        ps->arr[i] = i;
    } 

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

需要注意的是,arr和结构体是分别开辟空间的,结构体包含arr,所以释放的时候先释放arr,再释放结构体;如果先释放结构体,那么arr就找不到了。

那么这两种开辟空间的方式哪一种更好呢?第一种方案需要开辟、释放一次空间,第二种方案需要开辟、释放空间两次,后者更容易发生内存泄露,当malloc次数过多,那么内存碎片的数量就会越多,内存利用率会降低;换言之连续的内存存储有利于提高访问速度,减少内存碎片,提高内存利用率。

相关推荐
随意起个昵称4 小时前
区间dp-基础题目1(石子合并)
算法·动态规划
吞下星星的少年·-·4 小时前
线段树模板
算法
星空椰4 小时前
Python 面向对象高级:继承与类定义详解
开发语言·python
wunaiqiezixin5 小时前
如何在C++中创建和管理线程
c++
段一凡-华北理工大学5 小时前
2026 高炉炼铁智能化技术全景与演进路径~系列文章11:演进路径与行业未来
大数据·网络·人工智能·算法·工业智能体·高炉炼铁智能化
白露与泡影5 小时前
2026大厂Java面试题大全!牛客网最新版
java·开发语言
凯瑟琳.奥古斯特5 小时前
高阶子查询题目精炼
开发语言·数据库·python·职场和发展·数据库开发
喜欢踢足球的老罗5 小时前
从移动开发转型 AI Agent 工程师:我做了一个开源学习系统
人工智能·学习
雪度娃娃5 小时前
转向现代C++——在意为改写的函数添加 override
开发语言·c++
王老师青少年编程5 小时前
csp信奥赛C++高频考点专项训练之前缀和&差分 --【一维差分】:[NOIP 2018 提高组] 铺设道路
c++·前缀和·差分·csp·高频考点·信奥赛·铺设道路