c99柔性数组

变长结构体

1 struct test
2 {
3     int     nSize;
4     char    data[];    // 或者 char data[0];但建议使用 char data[]; 注意:c98 时不支持柔性数组,其仅作为非标准扩展。到c99时纳入标准
5 };

如上面代码即为一变长结构体,其中 char data[]; 为一变长数组,称之为柔性数组。正是因其为变长数组,故结构体才可变长。使用 test 结构体时,可用 malloc 申请大于 sizeof(test) 长度的空间。如下:

 1     const auto nSizeTest    = sizeof(test);
 2     const auto nExtraSize   = 3 * (sizeof(char));
 3
 4     test* stpTest           = (test*)malloc(nSizeTest + nExtraSize);
 5
 6     stpTest->nSize          = nSizeTest + nExtraSize;
 7     memset((void*)stpTest->data, 0x0L, nExtraSize);
 8     const auto nTempLen     = strlen(stpTest->data);
 9     for (auto nIndex = 0; nIndex < nExtraSize - 1; ++nIndex) {
10         stpTest->data[nIndex] = char(96 + nIndex);
11     }
12     stpTest->data[nExtraSize - 1] = ‘\0‘;
13
14     // some code here........
15
16     free(stpTest);          // 此处 free 的是刚才所申请的全部内存空间
17     stpTest = nullptr;

使用柔性数组有以下几个好处:

  • 首先柔性数组不占内存,值代表地址;
  • 可以通过stpTest->data来访问字符串,符合常规用法。
  • 字符串长度为动态分配。

关于柔性数组的注意点:

  1. 柔性数组只能放置于结构体的末尾声明
  2. 由于柔性数组是动态可变长的,则一般情况下只会用在没有继承关系、没有虚表的变长结构体(或类中)中,如果有继承关系的或虚表,则后果将非常严重。(有看过对象模型的人,肯定清楚为什么,此处就不多说)
  3. 上面变长结构体中的柔性数组除了 char 型别外,还可以是其他任何类型,甚至是自定义的类类型。但是:如果是自定义的类类型,则需要自己手动调用构造与析构。因为 malloc 与 free 时,不会自动调用构造与析构。从而可能导致不可预知的结果,以及内存泄漏的可能。比如:
 1 class obj
 2 {
 3 public:
 4     int nIndex;
 5     ~obj() {
 6         std::cout << "obj " << nIndex << " destroy" << std::endl;
 7     }
 8 };
 9
10 struct test
11 {
12     int nSize;
13     obj data[];
14 };
15
16 int _tmain(int argc, _TCHAR* argv[])
17 {
18
19     const auto nSizeTest = sizeof(test);
20     const auto nExtraSize = 3 * (sizeof(obj));
21
22     test* stpTest = (test*)malloc(nSizeTest + nExtraSize);
23
24     stpTest->nSize = nSizeTest + nExtraSize;
25     memset((void*)stpTest->data, 0x0L, nExtraSize);
26     for (auto nIndex = 0; nIndex < 3; ++nIndex) {
27         stpTest->data[nIndex].nIndex = nIndex + 1;
28     }
29
30     auto nSize = sizeof(test);
31     std::cout << "sizeof(test) = " << nSize << std::endl;       // 此处输出 4.说明柔性数组并不会占用 test 的空间.
32
33     nSize = sizeof(*stpTest);
34     std::cout << "sizeof(*stpTest) = " << nSize << std::endl;   // 此处输出 4
35
36     obj* ptr = stpTest->data;
37
38     free(stpTest);      // 此处将释放前面所申请的全部空间,但是:不会调用 ~obj() 函数!!!
39     stpTest = nullptr;
40
41     //ptr->nIndex = 777;  // 如果调用此句会蹦,因为前面 free 时已经全部释放掉所申请空间了,包括柔性数组 data 的那3个 obj 对象的空间
42
43     system("pause");
44
45     return 0;
46 }

参考文献:

时间: 2024-08-15 13:56:19

c99柔性数组的相关文章

结构体中最后一个成员为[0]或[1]长度数组(柔性数组成员)的用法

结构体中最后一个成员为[0]长度数组的用法:这是个广泛使用的常见技巧,常用来构成缓冲区.比起指针,用空数组有这样的优势:(1).不需要初始化,数组名直接就是所在的偏移:(2).不占任何空间,指针需要占用int长度空间,空数组不占任何空间.“这个数组不占用任何内存”,意味着这样的结构节省空间:“该数组的内存地址就和它后面的元素地址相同”,意味着无需初始化,数组名就是后面元素的地址,直接就能当指针使用. 这样的写法最适合制作动态buffer,因为可以这样分配空间malloc(sizeof(struc

柔性数组

柔性数组 * 柔性数组即数组大小待定的数组 * C语言中结构体的最后一个元素可以是大小未知的数组 * C语言中可以由结构体产生柔性数组 #include <stdio.h> #include <malloc.h> typedef struct _soft_array { int len; int array[]; }SoftArray; int main() { int i = 0; SoftArray* sa = (SoftArray*)malloc(sizeof(SoftArr

柔性数组成员 (flexible array member)-C99-ZZ

学习flexible array member是因为阅读Redis源码遇到的,sds.h中一开始就用到了. ============================================================================================== 在讲述柔性数组成员之前,首先要介绍一下不完整类型(incomplete type).不完整类型是这样一种类型,它缺乏足够的信息例如长度去描述一个完整的对象. 6.2.5 Types incomplete

柔性数组-读《深度探索C++对象模型》有感 (转载)

最近在看<深度探索C++对象模型>,对于Struct的用法中,发现有一些地方值得我们借鉴的地方,特此和大家分享一下,此间内容包含了网上搜集的一些资料,同时感谢提供这些信息的作者. 原文如下: 例如,把单一元素的数组放在一个struct的尾端,于是每个struct objects可以拥有可变大小的数组.    code: struct mumble { //stuff char pc[1];    };        //从档案或标准输入装置中取得一个字符串,然后为struct本身和该字符串配置

C中柔性数组(flexible array)的学习

首先看看什么是柔性数组 C99中,结构中的最后一个元素允许是未知大小的数组,这就叫做柔性数组成员,但结构中的柔性数组成员前面必须至少一个其 他成员.柔性数组成员允许结构中包含一个大小可变的数组.sizeof返回的这种结构大小不包括柔性数组的内存.包含柔性数组成员的结构用malloc ()函数进行内存的动态分配,并且分配的内存应该大于结构的大小,以适应柔性数组的预期大小. 基本模型为 typedef struct st_type { int nCnt; int item[]; }type_a; 在

柔性数组【转】

本文转载自:http://blog.csdn.net/renren900207/article/details/21227741 在讲述柔性数组成员之前,首先要介绍一下不完整类型(incomplete type).不完整类型是这样一种类型,它缺乏足够的信息例如长度去描述一个完整的对象. 6.2.5 Types incomplete types (types that describe objects but lack information needed to determine their s

C语言柔性数组

柔性数组:数组大小待定的数组.C语言中结构体最后一个元素可以是大小未知的数组.C语言可以由结构体产生柔性数组 柔性数组的结构如何只能堆上生成 柔性数组是C99的扩展,简而言之就是一个在struct结构里的标识占位符(不占结构struct的空间). #include <stdio.h> #include <malloc.h> typedef struct _soft_array { int len; int array[]; }SoftArray; int main() { int

柔性数组(结构体最后一个域为0/1数组)

结构体最后的长度为0或1数组的作用(转载) 2012-05-07 17:07:09 其实很 早在看LINUX下就看到这个东西,后来在MFC内存池里同样也看到了类似的东西,还依照MFC写过一个类似的小内存池,(MFC用的是return this + 1)后来在李先静的<系统程序员成长计划>里看到了类似的定义,于是心里想着总结一下,结果发现网上已经有牛人总结的很好了,于是乎就转了过来,谢谢你们 的分享,这是我前进的动力!同时,需要引起注意的:ISO/IEC 9899-1999里面,这么写是非法的,

第八章 柔性数组成员

在讲述柔性数组成员之前,首先要介绍一下不完整类型(incomplete type).不完整类型是这样一种类型,它缺乏足够的信息例如长度去描述一个完整的对象. 6.2.5 Types incomplete types (types that describe objects but lack information needed to determine their sizes). C与C++关于不完整类型的语义是一样的. 基本上没有什么书介绍过不完整类型,很多人初次遇到这个概念时脑袋会一片空白.