C/C++ 中长度为0的数组
參考文獻(xiàn):http://blog.csdn.net/zhaqiwen/article/details/7904515
近日在看項(xiàng)目中的框架代碼時(shí),發(fā)現(xiàn)了了一個(gè)奇特的語(yǔ)法:長(zhǎng)度為0的數(shù)組例如?
?uint8_t buf[0];?
我從未見(jiàn)過(guò)這樣的寫(xiě)法,所以在網(wǎng)上查了查資料,了解并記錄下來(lái).
在標(biāo)準(zhǔn)的C/C++中,長(zhǎng)度為0的數(shù)組是不被允許的,它算是一個(gè)C/C++擴(kuò)展,如果你的編譯器支持這個(gè)擴(kuò)展,你就可以使用它.
VS系列編譯器不完全支持這個(gè)擴(kuò)展,如果你這樣定義,多半會(huì)在編譯時(shí)出現(xiàn)這樣的警告:warning C4200: 使用了非標(biāo)準(zhǔn)擴(kuò)展 : 結(jié)構(gòu)/聯(lián)合中的零大小數(shù)組,當(dāng) UDT 包含大小為零的數(shù)組時(shí),無(wú)法生成復(fù)制構(gòu)造函數(shù)或副本賦值運(yùn)算符
GUN編譯器完全支持這個(gè)擴(kuò)展,你可以合法的聲明長(zhǎng)度為0的數(shù)組,但這種聲明的最典型的用法就是位于數(shù)組中的最后一項(xiàng),為了方便內(nèi)存緩沖區(qū)的管理,例如:
struct Line{uint32_t length;char contents[0]; };?在結(jié)構(gòu)體中,長(zhǎng)度為0的數(shù)組不會(huì)占用存儲(chǔ)空間?,在上述例子中?sizeof(Line)=4
在申請(qǐng)內(nèi)存空間時(shí),緩沖區(qū)的空間可以和結(jié)構(gòu)體的空間一起申請(qǐng),一次操作就可以完成.例如
uint32_t length = 10; struct Line *pLine = (struct Line *)malloc(sizeof (struct Line) + length); pLine->length = length;上述代碼就動(dòng)態(tài)地為結(jié)構(gòu)體申請(qǐng)了長(zhǎng)度(length)為10byte的緩沖區(qū),而且由于是同一次malloc操作,緩沖區(qū)與結(jié)構(gòu)體的內(nèi)存地址是連續(xù)的,而且可以按照數(shù)組下標(biāo)訪(fǎng)問(wèn)緩沖區(qū)元素,例如
for(uint32_t i = 0;i < pLine->length;++i) {pLine->contents[i] = i; }由于緩沖區(qū)與結(jié)構(gòu)體的內(nèi)存地址是連續(xù)的,在釋放內(nèi)存的時(shí)候,只需要一次free操作.
綜上所述,比起在結(jié)構(gòu)體中定義一個(gè)指針指向另一片緩沖區(qū)地址的做法,使用長(zhǎng)度為0的數(shù)組有以下好處:
1->指針本身需要占用內(nèi)存,而長(zhǎng)度為0的數(shù)組不需要
2->長(zhǎng)度為0的數(shù)組定義出的緩沖區(qū)可以和結(jié)構(gòu)體處在同一片連續(xù)地址中,只要一次malloc操作和free操作.如果用指針,需要分別申請(qǐng)和釋放結(jié)構(gòu)體內(nèi)存和指針指向的內(nèi)存塊,至少需要兩次以上的內(nèi)存操作.
測(cè)試代碼:
編譯器: gcc version 4.8.1 (tdm-2)
#include <stdio.h> #include <stdint.h> #include <malloc.h> struct Line{uint32_t length;uint8_t contents[0]; }; int32_t main() {uint32_t length = 10, i;printf("sizeof(Line)=%d\n", sizeof(struct Line));struct Line *pLine = (struct Line *)malloc(sizeof (struct Line) + length);pLine->length = length;for (i = 0; i < pLine->length; ++i){pLine->contents[i] = i;}for (i = 0; i < pLine->length; ++i){printf("i=%d,contents[i]=%d\n", i, pLine->contents[i]);}//free(pLine);return 0; }成功執(zhí)行并打印結(jié)果:
sizeof(Line)=4 i=0,contents[i]=0 i=1,contents[i]=1 i=2,contents[i]=2 i=3,contents[i]=3 i=4,contents[i]=4 i=5,contents[i]=5 i=6,contents[i]=6 i=7,contents[i]=7 i=8,contents[i]=8 i=9,contents[i]=9 Press any key to continue . . .在VS2013中編譯運(yùn)行上述代碼,除了會(huì)報(bào)警告 "?warning C4200 " 外,程序也可以正確運(yùn)行.
?
《新程序員》:云原生和全面數(shù)字化實(shí)踐50位技術(shù)專(zhuān)家共同創(chuàng)作,文字、視頻、音頻交互閱讀總結(jié)
以上是生活随笔為你收集整理的C/C++ 中长度为0的数组的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: linux下多路复用模型之Select模
- 下一篇: 孤儿进程与僵尸进程[总结]