青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

franksunny的個人技術(shù)空間
獲得人生中的成功需要的專注與堅持不懈多過天才與機會。 ——C.W. Wendte

由一道面試題來看 Struct 的對界

 

本文節(jié)選自宋寶華的C/C++struct深層探索一文,本人對其所描述的struct對齊比較喜歡,為此轉(zhuǎn)來與大家分享,原文見http://blog.donews.com/21cnbao/archive/2005/09/08/544877.aspx

 

Intel 、微軟等公司曾經(jīng)出過一道類似的面試題:

1. #include <iostream.h>

2. #pragma pack(8)

3. struct example1

4. {

5.     short a;

6.     long b;

7. };

8. struct example2

9. {

10.          char c;

11.          example1 struct1;

12.          short e;

13. };

14. #pragma pack()

 

15. int main(int argc, char* argv[])

16. {

17.          example2 struct2;

18.   cout << sizeof(example1) << endl;

19.   cout << sizeof(example2) << endl;

20.   cout << (unsigned int)(&struct2.struct1) - (unsigned int)(&struct2)

<< endl;

21. return 0;

22. }

問程序的輸入結(jié)果是什么?

答案是:

8

16

4

不明白?還是不明白?下面一一道來:

1 自然對界

struct 是一種復(fù)合數(shù)據(jù)類型,其構(gòu)成元素既可以是基本數(shù)據(jù)類型(如 int long float 等)的變量,也可以是一些復(fù)合數(shù)據(jù)類型(如 array struct union 等)的數(shù)據(jù)單元。對于結(jié)構(gòu)體,編譯器會自動進行成員變量的對齊,以提高運算效率。缺省情況下,編譯器為結(jié)構(gòu)體的每個成員按其自然對界( natural alignment )條件分配空間。各個成員按照它們被聲明的順序在內(nèi)存中順序存儲,第一個成員的地址和整個結(jié)構(gòu)的地址相同。

自然對界 (natural alignment) 即默認對齊方式,是指按結(jié)構(gòu)體的成員中 size 最大的成員對齊。

例如:

struct naturalalign

{

char a;

short b;

char c;

};

在上述結(jié)構(gòu)體中, size 最大的是 short ,其長度為 2 字節(jié),因而結(jié)構(gòu)體中的 char 成員 a c 都以 2 為單位對齊, sizeof(naturalalign) 的結(jié)果等于 6

如果改為:

struct naturalalign

{

char a;

int b;

char c;

};

其結(jié)果顯然為 12

 

2 指定對界

一般地,可以通過下面的方法來改變?nèi)笔〉膶鐥l件:

· 使用偽指令 #pragma pack (n) ,編譯器將按照 n 個字節(jié)對齊;

· 使用偽指令 #pragma pack () ,取消自定義字節(jié)對齊方式。

注意:如果 #pragma pack (n) 中指定的 n 大于結(jié)構(gòu)體中最大成員的 size ,則其不起作用,結(jié)構(gòu)體仍然按照 size 最大的成員進行對界。

例如:

#pragma pack (n)

struct naturalalign

{

char a;

int b;

char c;

};

#pragma pack ()

當(dāng) n 4 8 16 時,其對齊方式均一樣, sizeof(naturalalign) 的結(jié)果都等于 12 。而當(dāng) n 2 時,其發(fā)揮了作用,使得 sizeof(naturalalign) 的結(jié)果為 8

VC++ 6.0 編譯器中,我們可以指定其對界方式(見圖 1 ),其操作方式為依次選擇 projetct > setting > C/C++ 菜單,在 struct member alignment 中指定你要的對界方式。

1  在 VC++ 6.0 中指定對界方式

另外,通過 __attribute((aligned (n))) 也可以讓所作用的結(jié)構(gòu)體成員對齊在 n 字節(jié)邊界上,但是它較少被使用,因而不作詳細講解。

 

3   面試題的解答

至此,我們可以對 Intel 、微軟的面試題進行全面的解答。

程序中第 2 #pragma pack (8) 雖然指定了對界為 8 ,但是由于 struct example1 中的成員最大 size 4 long 變量 size 4 ),故 struct example1 仍然按 4 字節(jié)對界, struct example1 size 8 ,即第 18 行的輸出結(jié)果;

struct example2 中包含了 struct example1 ,其本身包含的簡單數(shù)據(jù)成員的最大 size 2 short 變量 e ),但是因為其包含了 struct example1 ,而 struct example1 中的最大成員 size 4 struct example2 也應(yīng)以 4 對界, #pragma pack (8) 中指定的對界對 struct example2 也不起作用,故 19 行的輸出結(jié)果為 16

由于 struct example2 中的成員以 4 為單位對界,故其 char 變量 c 后應(yīng)補充 3 個空,其后才是成員 struct1 的內(nèi)存空間, 20 行的輸出結(jié)果為 4

 

 

在閱讀了此節(jié)之后,本人對為什么在使用過程中用sizeof取得的結(jié)構(gòu)長度不同的問題有了認識,不過本人調(diào)試時,不光跟對界方式有關(guān),還和設(shè)置的Processor有關(guān),這方面還請大俠幫我指點下迷津。

    過了快一年了,回過頭來再看看這篇文章發(fā)現(xiàn)還是有用的,另外再加上一句別人:當(dāng)未用 #pragma 指令指定編譯器的對齊位數(shù)時,結(jié)構(gòu)體按最長寬度的數(shù)據(jù)成員的寬度對齊;當(dāng)使用了#pragma 指令指定編譯器的對齊位數(shù)時,結(jié)構(gòu)體按最長寬度的數(shù)據(jù)成員的寬度和 #pragma 指令指定的位數(shù)中的較小值對齊。(2007年10月8日) 

posted on 2006-10-20 21:52 frank.sunny 閱讀(2150) 評論(9)  編輯 收藏 引用 所屬分類: C/C++學(xué)習(xí)和實踐

FeedBack:
# re: 由一道面試題來看Struct的對界(再談結(jié)構(gòu))
2006-12-22 13:32 | mumutou
錯誤,struct所說的對齊不是這么回事
  回復(fù)  更多評論
  
# re: 由一道面試題來看Struct的對界(再談結(jié)構(gòu))
2006-12-22 13:34 | mumutou
char a;

int b;

char c;
按照4邊界對齊,1+4+1 =6;
(6/4+1) * 4 = 8
也就是說對struct總的長度進行4邊界對齊,而不是對每個struct內(nèi)容4邊界對齊  回復(fù)  更多評論
  
# re: 由一道面試題來看Struct的對界(再談結(jié)構(gòu))
2006-12-26 20:59 | frank.sunny
謝謝樓上的大俠提點,在你的指點下,認真看了下對齊的東西,你說的也不是很對
上面的題按4邊界對齊的話,也要考慮順序的即 1+(3)+ 4 + 1 = 9
(9 / 4 +1) * 4 = 12
也就前一個類型根據(jù)后一個類型擴展
具體我再附上別人的源代碼。

struct DATA1
{
char c1; //偏移量0,累積size = 1
char c2; //偏移量1,累積size = 1 + 1 = 2
short si; //偏移量2,累積size = 2 + 2
};

struct DATA2
{
char c1; //偏移量0,累積size = 1
short si; //偏移量1 + (1),累積size = 1 + (1) + 2 = 4
char c2; //偏移量4,累積size = 4 + 1 = 5,但按最大長度sizeof(short) = 2對齊,故最后取6
};

struct DATA3
{
char c1; //偏移量0,累積size = 1
double d; //偏移量1 + (7),累積size = 1 + (7) + 8 = 16
char c2; //偏移量16,累積size = 16 + 1 = 17,但按最大長度sizeof(double) = 8對齊,故最后取24
};

#pragma pack(push,1) //強制1字節(jié)對齊
struct DATA4
{
char c1; //偏移量0,累積size = 1
double d; //偏移量1,累積size = 1 + 8 = 9
char c2; //偏移量9,累積size = 9 + 1 = 10
};
#pragma pack(pop) //恢復(fù)默認對齊方式

struct DATA5
{
char c1;
double d;
char c2;
};

void main()
{
cout << "sizeof(DATA1) = " << sizeof(DATA1) << endl;
cout << "sizeof(DATA2) = " << sizeof(DATA2) << endl;
cout << "sizeof(DATA3) = " << sizeof(DATA3) << endl;
cout << "sizeof(DATA4) = " << sizeof(DATA4) << endl;
cout << "sizeof(DATA5) = " << sizeof(DATA5) << endl;
}  回復(fù)  更多評論
  
# re: 由一道面試題來看Struct的對界(再談結(jié)構(gòu))
2008-07-18 11:37 | dshy
struct test {
char x1;
short x2;
float x3;
char x4;
};

sizeof(test)是多少?  回復(fù)  更多評論
  
# re: 由一道面試題來看Struct的對界(再談結(jié)構(gòu))
2009-03-04 17:59 | JYSG3
錯誤,誤人子弟  回復(fù)  更多評論
  
# re: 由一道面試題來看Struct的對界(再談結(jié)構(gòu))
2009-03-18 16:12 | 撒旦
sizeof(test)是多少?
//應(yīng)該是12個字節(jié)吧,1 + 2 + (1) + 4 +1 =9,因為是以4字節(jié)對起,所以應(yīng)該是12字節(jié)。  回復(fù)  更多評論
  
# re: 由一道面試題來看Struct的對界(再談結(jié)構(gòu))
2010-04-24 11:43 | 小林
您好: 我讀了你的代碼分析,收獲頗多。也發(fā)現(xiàn)了一點小問題,現(xiàn)向你提一下。
我將你的程序在機子上運行,發(fā)現(xiàn)你的分析結(jié)果跟電腦顯示的結(jié)果不一樣。
double,long 最大長度不是按8 對齊,而是 按4對齊。
struct DATA3
{
char c1; //偏移量0,累積size = 1
double d; //偏移量1 + (3),累積size = 1 + (3) + 8 = 112
char c2; //偏移量16,累積size = 12+ 1 = 13,但按最大長度sizeof(double) = 4對齊,故最后取16
};   回復(fù)  更多評論
  
# re: 由一道面試題來看Struct的對界(再談結(jié)構(gòu))
2011-03-16 13:53 | casper
路過,學(xué)習(xí),有問題:
struct DATA3
{
char c1; //偏移量0,累積size = 1
double d; //偏移量1 + (7),累積size = 1 + (7) + 8 = 16
char c2; //偏移量16,累積size = 16 + 1 = 17,但按最大長度sizeof(double) = 8對齊,故最后取24
};
就拿這個來說 這個的結(jié)果應(yīng)該是16

不能用單獨的最長來界定

單個的元素需要在結(jié)構(gòu)體內(nèi)部對齊

同時最后還需要界定整個結(jié)構(gòu)體的對齊問題  回復(fù)  更多評論
  
# re: 由一道面試題來看Struct的對界(再談結(jié)構(gòu))
2011-03-16 14:03 | casper
struct DATA3{
char c1;
double d;
char c2;
};
*結(jié)構(gòu)體內(nèi)部 有補齊需求的只有d
sizeof(double) > 4 ---> 4
c1 block 1+(3)=4
total: 1+(3)+8+1=13
*結(jié)構(gòu)體開始對齊
sizeof(double) > 4 ---> 4 4*4-3=13
c2 block 1+(3)=4
total: 1+(3)+8+1+(3)=16@casper
  回復(fù)  更多評論
  

常用鏈接

留言簿(13)

隨筆分類

個人其它博客

基礎(chǔ)知識鏈接

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            久久精品青青大伊人av| 欧美怡红院视频一区二区三区| 国产亚洲欧美另类中文| 亚洲精品专区| 亚洲国产精品欧美一二99| 亚洲欧美国产高清va在线播| 一本大道久久a久久综合婷婷| 久久亚洲不卡| 久久人人爽人人爽| 国产欧美日韩精品一区| av成人国产| 一区二区欧美激情| 欧美日韩国产va另类| 欧美国产先锋| 亚洲国产天堂久久国产91| 久久丁香综合五月国产三级网站| 性伦欧美刺激片在线观看| 欧美日本在线观看| 91久久综合| 亚洲区国产区| 欧美激情精品久久久六区热门 | 亚洲一区二区三区免费在线观看| 美女久久一区| 欧美福利电影网| 91久久精品国产91性色tv| 久久免费视频网| 欧美激情亚洲自拍| 亚洲精品中文字幕在线| 欧美理论片在线观看| 亚洲啪啪91| 亚洲一级在线观看| 国产精品视频网站| 亚洲在线中文字幕| 久久精品三级| 亚洲国产二区| 欧美日韩成人精品| 在线亚洲一区观看| 欧美一级二区| 一区免费观看视频| 欧美激情视频网站| 亚洲精品一级| 久久成人这里只有精品| 国语自产精品视频在线看一大j8| 久久婷婷蜜乳一本欲蜜臀| 欧美激情在线观看| 亚洲无限av看| 国产主播在线一区| 美女黄色成人网| 99国产一区| 久久久一二三| 日韩视频一区二区三区在线播放| 国产精品hd| 欧美综合国产| 国产乱码精品一区二区三区五月婷| 亚洲综合视频在线| 国产原创一区二区| 免费观看日韩av| 欧美国产精品专区| 久久经典综合| 欧美在线看片| 欧美在线啊v一区| 亚洲欧美精品在线| 亚洲午夜精品| 亚洲一区二区三区乱码aⅴ蜜桃女 亚洲一区二区三区乱码aⅴ | 亚洲欧美激情视频| 亚洲天堂成人在线观看| 中文网丁香综合网| 亚洲免费观看高清完整版在线观看| 欧美激情一区二区三区成人| 欧美国产国产综合| 欧美a级一区| 欧美v日韩v国产v| 欧美成人一区二区三区在线观看 | 亚洲免费在线观看视频| 这里只有视频精品| 亚洲欧美高清| 欧美一区二区三区在线播放| 欧美伊人精品成人久久综合97| 欧美呦呦网站| 老司机一区二区三区| 欧美成人性生活| 欧美日本一道本| 国产精品盗摄久久久| 国产精品视频免费| 国外成人免费视频| 亚洲国产精品www| 99精品国产在热久久下载| 亚洲伊人网站| 久久精品91久久香蕉加勒比| 欧美77777| 亚洲毛片一区| 欧美亚洲视频一区二区| 麻豆精品一区二区av白丝在线| 欧美激情一级片一区二区| 国产精品福利在线观看网址| 国产一区二区三区直播精品电影 | 亚洲欧美日韩精品久久久| 久久久av水蜜桃| 欧美激情1区| 国产精品欧美日韩久久| 精品福利免费观看| av成人动漫| 久久大综合网| 亚洲欧洲精品一区| 亚洲午夜久久久久久久久电影院 | 国产精品日韩二区| 伊人久久久大香线蕉综合直播| 99国产精品久久久| 久久福利影视| 亚洲第一在线综合在线| 亚洲无线观看| 免费人成精品欧美精品| 国产精品日产欧美久久久久| 亚洲国产精品一区二区三区| 亚洲女性裸体视频| 欧美大秀在线观看| 亚洲欧美日韩精品久久| 欧美精品粉嫩高潮一区二区| 国内精品久久久久久影视8 | 国产一区二区0| 亚洲视频一区在线观看| 美女图片一区二区| 在线视频欧美日韩精品| 免费在线日韩av| 国产一二三精品| 亚洲婷婷免费| 亚洲丰满少妇videoshd| 欧美一区二区三区四区在线| 欧美三级免费| 亚洲精品一区二区三区99| 久久精品亚洲国产奇米99| 夜夜精品视频一区二区| 欧美成人日本| 影音先锋久久久| 久久精品日韩| 亚洲永久在线观看| 欧美日韩国产色站一区二区三区| 亚洲国产二区| 噜噜噜噜噜久久久久久91| 亚洲自拍偷拍一区| 欧美午夜不卡在线观看免费 | 国产欧美一区二区精品忘忧草| 一区二区三区高清视频在线观看| 欧美电影在线播放| 久久久99久久精品女同性| 国产日韩欧美综合一区| 香蕉成人啪国产精品视频综合网| 亚洲美女在线看| 欧美欧美天天天天操| 亚洲免费成人av电影| 亚洲国产成人av好男人在线观看| 久久久久99| 在线观看的日韩av| 麻豆91精品91久久久的内涵| 久久不见久久见免费视频1| 国产欧美不卡| 久久精品盗摄| 欧美一区二区免费| 国产一区二区高清不卡| 久久久伊人欧美| 久久久久久久97| 在线成人av| 欧美激情片在线观看| 欧美成人精品1314www| 亚洲欧洲综合另类| 亚洲区中文字幕| 欧美日韩成人精品| 亚洲一区二区少妇| 亚洲香蕉网站| 国产午夜精品在线| 美玉足脚交一区二区三区图片| 久久久久综合一区二区三区| 亚洲国产精品一区二区三区| 亚洲黄色成人| 欧美特黄a级高清免费大片a级| 亚洲欧美乱综合| 欧美资源在线观看| 永久免费视频成人| 亚洲国产精品va在线看黑人| 欧美日韩午夜剧场| 午夜精品在线看| 久久精品动漫| 99日韩精品| 亚洲尤物视频网| 在线视频国产日韩| 亚洲人成7777| 国产精品一二一区| 欧美va亚洲va香蕉在线| 欧美日韩精品一本二本三本| 午夜亚洲激情| 老司机aⅴ在线精品导航| 一本色道久久综合亚洲精品按摩 | 美国成人直播| 亚洲一级片在线观看| 欧美中文字幕在线播放| 亚洲精品午夜精品| 亚洲欧美日韩综合一区| 亚洲国产专区| 亚洲男女自偷自拍| 亚洲激情一区|