新聞中心

EEPW首頁(yè) > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > 漫談c語言結(jié)構(gòu)體

漫談c語言結(jié)構(gòu)體

作者: 時(shí)間:2016-12-01 來源:網(wǎng)絡(luò) 收藏

}

#includeint main(){struct	  //聲明結(jié)構(gòu)體char_short_long{char  c;short s;long  l;}char_short_long;struct	  //聲明結(jié)構(gòu)體long_short_char{long  l;short s;char  c;}long_short_char;struct	  //聲明結(jié)構(gòu)體char_long_short{char  c;long  l;short s;}char_long_short;printf(" ");printf(" Size of char   = %d bytes",sizeof(char));printf(" Size of shrot  = %d bytes",sizeof(short));printf(" Size of long   = %d bytes",sizeof(long));printf(" ");	//char_short_longprintf(" Size of char_short_long       = %d bytes",sizeof(char_short_long));printf("     Addr of char_short_long.c = 0x%p (10進(jìn)制:%d)",&char_short_long.c,&char_short_long.c);printf("     Addr of char_short_long.s = 0x%p (10進(jìn)制:%d)",&char_short_long.s,&char_short_long.s);printf("     Addr of char_short_long.l = 0x%p (10進(jìn)制:%d)",&char_short_long.l,&char_short_long.l);printf(" ");printf(" ");	//long_short_charprintf(" Size of long_short_char       = %d bytes",sizeof(long_short_char));printf("     Addr of long_short_char.l = 0x%p (10進(jìn)制:%d)",&long_short_char.l,&long_short_char.l);printf("     Addr of long_short_char.s = 0x%p (10進(jìn)制:%d)",&long_short_char.s,&long_short_char.s);printf("     Addr of long_short_char.c = 0x%p (10進(jìn)制:%d)",&long_short_char.c,&long_short_char.c);printf(" ");printf(" ");	//char_long_shortprintf(" Size of char_long_short       = %d bytes",sizeof(char_long_short));printf("     Addr of char_long_short.c = 0x%p (10進(jìn)制:%d)",&char_long_short.c,&char_long_short.c);printf("     Addr of char_long_short.l = 0x%p (10進(jìn)制:%d)",&char_long_short.l,&char_long_short.l);printf("     Addr of char_long_short.s = 0x%p (10進(jìn)制:%d)",&char_long_short.s,&char_long_short.s);printf(" ");return 0;}


程序的運(yùn)行結(jié)果如下(注意:括號(hào)內(nèi)的數(shù)據(jù)是成員變量的地址的十進(jìn)制形式):

2.結(jié)構(gòu)體成員變量?jī)?nèi)存對(duì)齊

首先,我們來分析一下上面程序的運(yùn)行結(jié)果。前三行說明在我的程序中,char型占1個(gè)字節(jié),short型占2個(gè)字節(jié),long型占4個(gè)字節(jié)。char_short_long、long_short_char和char_long_short是三個(gè)結(jié)構(gòu)體成員相同但是成員變量的排列順序不同。并且從程序的運(yùn)行結(jié)果來看,

Size of char_short_long =8bytes
Sizeof long_short_char=8bytes
Sizeof char_long_short=12bytes//比前兩種情況大4byte!

并且,還要注意到,1 byte (char)+2byte(short)+4 byte(long) = 7 byte,而不是8 byte。

所以,結(jié)構(gòu)體成員變量的放置順序影響著結(jié)構(gòu)體所占的內(nèi)存空間的大小。一個(gè)結(jié)構(gòu)體變量所占內(nèi)存的大小不一定等于其成員變量所占空間之和。如果一個(gè)用戶程序或者操作系統(tǒng)(比如uC/OS-II)中存在大量結(jié)構(gòu)體變量時(shí),這種內(nèi)存占用必須要進(jìn)行優(yōu)化,也就是說,結(jié)構(gòu)體內(nèi)部成員變量的排列次序是有講究的。

結(jié)構(gòu)體成員變量到底是如何存放的呢?

在這里,我就不賣關(guān)子了,直接給出如下結(jié)論,在沒有#pragmapack宏的情況下:

原則1結(jié)構(gòu)(struct或聯(lián)合union)的數(shù)據(jù)成員,第一個(gè)數(shù)據(jù)成員放在offset為0的地方,以后每個(gè)數(shù)據(jù)成員存儲(chǔ)的起始位置要從該成員大小的整數(shù)倍開始(比如int在32位機(jī)為4字節(jié),則要從4的整數(shù)倍地址開始存儲(chǔ))。

原則2結(jié)構(gòu)體的總大小,也就是sizeof的結(jié)果,必須是其內(nèi)部最大成員的整數(shù)倍,不足的要補(bǔ)齊。

*原則3結(jié)構(gòu)體作為成員時(shí),結(jié)構(gòu)體成員要從其內(nèi)部最大元素大小的整數(shù)倍地址開始存儲(chǔ)。(structa里存有structb,b里有char,int,double等元素時(shí),那么b應(yīng)該從8的整數(shù)倍地址處開始存儲(chǔ),因?yàn)閟izeof(double)=8bytes)

這里,我們結(jié)合上面的程序來分析(暫時(shí)不討論原則3)。

先看看char_short_long和long_short_char這兩個(gè)結(jié)構(gòu)體,從它們的成員變量的地址可以看出來,這兩個(gè)結(jié)構(gòu)體符合原則1和原則2。注意,在char_short_long的成員變量的地址中,char_short_long.s的地址是1244994,也就是說,1244993是“空的”,只是被“占位”了!



關(guān)鍵詞: c語言結(jié)構(gòu)

評(píng)論


技術(shù)專區(qū)

關(guān)閉