美文网首页
柔性数组

柔性数组

作者: GOGOYAO | 来源:发表于2019-08-10 15:27 被阅读0次

在接触变长结构体之前,以为会是一个很难理解的东西,但是这其实算是C里面的一种技巧吧,优点是:分配了一段连续的内存,防止内存碎片化以及方便内存的管理。

使用变长结构体的格式如下:

struct Test
{
  ....
  int a;
  ....
  char b[0];
};

重点是结构体的最后一个成员char b[0],是个空数组。在我们不知道结构体内的某个成员大小是多少的时候,我们在最后一个成员放置了一个空数组,这样做的好处就是,我们直接用结构体指针申请空间(sizeof(struct) + 给空数组申请的空间),就完成了动态分配。

这里可能读者会有一个疑惑,那就是指针也同样完成这个任务,为什么不用指针呢。这里我们就用一个例子来说明:

#include <iostream>
using namespace std;
#include <cstring>
#include <stdlib.h>
typedef struct
{
    int a;
    char b[0];
}Empty;
typedef struct
{
    int x;
    char *y;
}Ptr;
int main()
{
    //变长结构体申请内存
    Empty *p_array = (Empty *)malloc(sizeof(Empty) + 100);  //可以看到这里直接申请了变长结构体大小的空间再加100字节,这100字节就相当于是给char b[0]用的了
    //常规方式申请内存,先申请结构体的,再申请指针的
    Ptr *p_ptr = (Ptr *)malloc(sizeof(Ptr));
    p_ptr->y = (char *)malloc(100);
    //初始化malloc申请的内存
    memset(p_array, 0, sizeof(p_array));
    memset(p_ptr, 0, sizeof(p_ptr));
    memset(p_ptr->y, 0, 100);


    //拷贝字符串并输出
    strcpy(p_array->b, "test");
    strcpy(p_ptr->y, "test");
    cout << p_array->b << endl;
    cout << p_ptr->y << endl;

    //释放空间
    cout << "--------------------------" << endl;
    if(p_array != NULL)
    {
        cout << "1.释放变长结构体的空间" << endl;
        free(p_array);
        p_array = NULL;
    }
    cout << "--------------------------" << endl;
    if(p_ptr->y != NULL)
    {
        cout << "1.释放指针申请的空间" << endl;
        free(p_ptr->y);
        p_ptr->y = NULL;
        if(p_ptr != NULL)
        {
            cout << "2.释放结构体的空间" << endl;
            free(p_ptr);
            p_ptr = NULL;
        }
    }
    return 0;
}

两者相比较,得出以下主要结论:
变长结构体的内存是连续的(严谨的说是虚拟内存),而常规方法的不是,所以变长结构体只需释放一次空间,而常规方法需要释放两次。

并且我们使用sizeof查看两个结构体的大小会发现,Empty结构体的大小为4字节,而Ptr的大小为8字节,可以得出char b[0]是不占内存空间的。

最后还有需要注意的一点就是,有些编译器可能不支持char b[0]这样的写法,可以换成char b[1],用法还是一样的,只是这时的结构体大小变成了4 + 1 + 3 = 8字节,最后那个3字节是由于内存对齐填补上的。

相关文章

  • C++中的柔性数组

    描述 C99标准提供在结构体最后一个元素定义一个未知大小的数组,它就叫做柔性数组成员。 柔性数组的特点 定义柔性数...

  • 柔性数组

    在接触变长结构体之前,以为会是一个很难理解的东西,但是这其实算是C里面的一种技巧吧,优点是:分配了一段连续的内存,...

  • C 实现链表的高速公路-跳跃表

    本文所需基础: 1.知道如何用c写链表。 2.下文将会教到的柔性数组。 预准备(柔性数组): 为了下文讲述方便,先...

  • 可变长数组的实现

    以前从来没有用过柔性数组,今天学习一下。柔性数组也就是指char a[0] 或者 char a[]基本方式如下 其...

  • C语言中的不定长数组----柔性数组

    柔性数组不占用struct的空间, 而是额外分配的空间

  • 零长度数组与柔性数组

    零长度数组(Zero-length array)与柔性数组(Flexible array) GNU C允许使用零长...

  • C柔性数组和变长数组

    变长数组 C语言在C99以前,数组的维度必须是整数常量表达式, 而C99则做了很大改进,允许数组维度为整形变量或者...

  • C 语言之柔性数组

    一 历史 在c99标准出来之前。如果要在某个结构体中使用字符串变时,为了使字符串变量存储地址能与结构体整体连在一起...

  • C语言中的柔性数组

    C语言中 结构体末尾如果是定义的数组,那么这个数组只是一个标识符 并不占用结构体大小。 我们可以在结构体中定义一个...

  • C 迷你系列(二)柔性数组

    引言 当我们阅读 Redis 源码的时候,发现有些结构体会有如下一个数组元素: 思考一下的话,你会发现,我们在声明...

网友评论

      本文标题:柔性数组

      本文链接:https://www.haomeiwen.com/subject/kfyrjctx.html