GNU C中的零長(zhǎng)數(shù)組

C語(yǔ)言沒(méi)有STL,缺乏對(duì)動(dòng)態(tài)長(zhǎng)度字符串功能的支持钻蹬;同時(shí)吼蚁,C使用'\0'判斷字符串的結(jié)尾,不具有二進(jìn)制安全性。然而在程序開(kāi)發(fā)時(shí)肝匆,我們可以通過(guò)自定義的方式粒蜈,實(shí)現(xiàn)簡(jiǎn)易的動(dòng)態(tài)長(zhǎng)度字符串功能。
動(dòng)態(tài)長(zhǎng)度字符串結(jié)構(gòu)定義如下:

struct DynamicBuffer
{
  unsigned len;  // 實(shí)際長(zhǎng)度
  unsigned free;  // 剩余可用空間大小
  char data[0];  // 實(shí)際內(nèi)容
};

其中旗国,較為少見(jiàn)的是DynamicBuffer結(jié)構(gòu)中的最后一項(xiàng)——定義了長(zhǎng)度為0的字符數(shù)組枯怖。零長(zhǎng)數(shù)組是在GNU C中定義的一項(xiàng)功能:

Zero-length arrays are allowed in GNU C. They are very useful as the last element of a structure which is really a header for a variable-length object.

在其他環(huán)境中編譯含有零長(zhǎng)數(shù)組的代碼可能會(huì)導(dǎo)致編譯錯(cuò)誤。

在DynamicBuffer的定義中能曾,我們當(dāng)然也可以用char*來(lái)保存字符串?dāng)?shù)據(jù)度硝。但是,使用零長(zhǎng)數(shù)組代替數(shù)組指針具有以下優(yōu)勢(shì):

  • 指針占用存儲(chǔ)空間寿冕;而零長(zhǎng)數(shù)組則不占用結(jié)構(gòu)體空間蕊程。
  • 使用指針需要初始化,使用時(shí)需要間接尋址驼唱;零長(zhǎng)數(shù)組不需要初始化存捺,內(nèi)存地址和后面的元素地址相同,數(shù)組名可以直接當(dāng)作指針使用曙蒸。

程序測(cè)試用例如下:

#include <stdio.h>
#include <stdlib.h>
#define LENGTH 5

struct DynamicBuffer1
{
    unsigned len;
    unsigned free;
    char *data;
}__attribute((packed));

struct DynamicBuffer2
{
    unsigned len;
    unsigned free;
    char data[0];
}__attribute((packed));

int main()
{
    struct DynamicBuffer1 *b1;
    struct DynamicBuffer2 *b2;
    int i;

    printf("DynamicBuffer1 length: %d\n",sizeof(struct DynamicBuffer1));
    printf("DynamicBuffer2 length: %d\n",sizeof(struct DynamicBuffer2));

    b1=(struct DynamicBuffer1*)malloc(sizeof(struct DynamicBuffer1));
    b1->len=0;
    b1->free=LENGTH;
    b1->data=(char *)malloc(sizeof(char)*LENGTH + 1);
    b1->data[0] = 100;
    b1->free--;
    b1->len++;
    printf("\nb1->len=%d,b1->free=%d,*(b1->data)=%d\n",b1->len,b1->free,*b1->data);
    printf("b1\t\t address: %p\n", b1);
    printf("b1->len\t address: %p\n", &(b1->len));
    printf("b1->free\t address: %p\n", &(b1->free));
    printf("b1->data\t address: %p\n", b1->data);

    b2=(struct DynamicBuffer2*)malloc(sizeof(struct DynamicBuffer2)+sizeof(char)*LENGTH + 1);
    b2->len=0;
    b2->free=LENGTH;
    for(i=0;i<LENGTH;i++)
    {
        b2->data[i]=i;
        b2->free--;
        b2->len++;
    }
    printf("\nb2->len=%d,b2->free=%d\n",b2->len,b2->free);
    printf("b2\t\taddress: %p\n", b2);
    printf("b2->len\taddress: %p\n", &(b2->len));
    printf("b2->free\taddress: %p\n", &(b2->free));
    printf("b2->data\taddress: %p\n", b2->data);
    free(b1->data);
    free(b1);
    free(b2);
}

用例輸出:

DynamicBuffer1 length: 16
DynamicBuffer2 length: 8
b1->len=1,b1->free=4,*(b1->data)=100
b1 address: 0x1ae7010
b1->len address: 0x1ae7010
b1->free address: 0x1ae7014
b1->data address: 0x1ae7030
b2->len=5,b2->free=0
b2 address: 0x1ae7050
b2->len address: 0x1ae7050
b2->free address: 0x1ae7054
b2->data address: 0x1ae7058

可見(jiàn)捌治,通過(guò)零長(zhǎng)數(shù)組定義的DynamicBuffer2占用的內(nèi)存空間只有8個(gè)字節(jié)(兩個(gè)整形變量的空間),且data數(shù)組地址緊鄰DynamicBuffer2的內(nèi)存地址纽窟。Redis的SDS API也使用了類似的方式肖油,提供了二進(jìn)制安全的動(dòng)態(tài)長(zhǎng)度字符串功能。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末臂港,一起剝皮案震驚了整個(gè)濱河市森枪,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌审孽,老刑警劉巖县袱,帶你破解...
    沈念sama閱讀 219,427評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異佑力,居然都是意外死亡式散,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,551評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)打颤,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)暴拄,“玉大人,你說(shuō)我怎么就攤上這事编饺」耘瘢” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 165,747評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵透且,是天一觀的道長(zhǎng)撕蔼。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么鲸沮? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,939評(píng)論 1 295
  • 正文 為了忘掉前任畅形,我火速辦了婚禮,結(jié)果婚禮上诉探,老公的妹妹穿的比我還像新娘日熬。我一直安慰自己,他們只是感情好肾胯,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,955評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布竖席。 她就那樣靜靜地躺著,像睡著了一般敬肚。 火紅的嫁衣襯著肌膚如雪毕荐。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,737評(píng)論 1 305
  • 那天艳馒,我揣著相機(jī)與錄音憎亚,去河邊找鬼。 笑死弄慰,一個(gè)胖子當(dāng)著我的面吹牛第美,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播陆爽,決...
    沈念sama閱讀 40,448評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼什往,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了慌闭?” 一聲冷哼從身側(cè)響起别威,我...
    開(kāi)封第一講書(shū)人閱讀 39,352評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎驴剔,沒(méi)想到半個(gè)月后省古,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,834評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡丧失,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,992評(píng)論 3 338
  • 正文 我和宋清朗相戀三年豺妓,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片利花。...
    茶點(diǎn)故事閱讀 40,133評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡科侈,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出炒事,到底是詐尸還是另有隱情,我是刑警寧澤蔫慧,帶...
    沈念sama閱讀 35,815評(píng)論 5 346
  • 正文 年R本政府宣布挠乳,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏睡扬。R本人自食惡果不足惜盟蚣,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,477評(píng)論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望卖怜。 院中可真熱鬧屎开,春花似錦、人聲如沸马靠。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,022評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)甩鳄。三九已至逞度,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間妙啃,已是汗流浹背档泽。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,147評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留揖赴,地道東北人馆匿。 一個(gè)月前我還...
    沈念sama閱讀 48,398評(píng)論 3 373
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像燥滑,于是被迫代替她去往敵國(guó)和親甜熔。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,077評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容