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

  C++博客 :: 首頁 :: 聯(lián)系 ::  :: 管理
  163 Posts :: 4 Stories :: 350 Comments :: 0 Trackbacks

常用鏈接

留言簿(48)

我參與的團(tuán)隊

搜索

  •  

積分與排名

  • 積分 - 402332
  • 排名 - 59

最新評論

閱讀排行榜

評論排行榜


    首先請大家先看下面代碼:
    typedef struct
    {
      UINT32  NumElements;
      union
      {
         UINT32  ObjectHandle;
       }Entry;
     }STR_ARRAY, *PSTR_ARRAY;

    還有這兩句#pragma pack(push, 1)
    #pragma pack(pop)
    #pragma  pack( [ n ] )

    該指令指定結(jié)構(gòu)和聯(lián)合成員的緊湊對齊。而一個完整的轉(zhuǎn)換單元的結(jié)構(gòu)和聯(lián)合的緊湊對齊由/ Z p 選項設(shè)置。緊湊對齊用p a c e 編譯指示在數(shù)據(jù)說明層設(shè)置。該編譯指示在其出現(xiàn)后的第一個結(jié)構(gòu)或聯(lián)合說明處生效。該編譯指示對定義無效。當(dāng)你使用#pragma  pack ( n ) 時, 這里n 為1 、2 、4 、8 或1 6 。第一個結(jié)構(gòu)成員之后的每個結(jié)構(gòu)成員都被存儲在更小的成員類型或n 字節(jié)界限內(nèi)。如果你使用無參量的#pragma  pack , 結(jié)構(gòu)成員被緊湊為以/ Z p 指定的值。該缺省/ Z p 緊湊值為/ Z p 8 。

    編譯器也支持以下增強(qiáng)型語法:
    #pragma  pack( [ [ { p u s h | p o p } , ] [ 標(biāo)識符, ] ] [ n] )若不同的組件使用p a c k 編譯指示指定不同的緊湊對齊, 這個語法允許你把程序組件組合為一個單獨的轉(zhuǎn)換單元。帶p u s h 參量的p a c k 編譯指示的每次出現(xiàn)將當(dāng)前的緊湊對齊存儲到一個內(nèi)部編譯器堆棧中。編譯指示的參量表從左到右讀取。如果你使用p u s h , 則當(dāng)前緊湊值被存儲起來; 如果你給出一個n 的值, 該值將成為新的緊湊值。若你指定一個
標(biāo)識符, 即你選定一個名稱, 則該標(biāo)識符將和這個新的的緊湊值聯(lián)系起來。帶一個p o p 參量的p a c k 編譯指示的每次出現(xiàn)都會檢索內(nèi)部編譯器堆棧頂?shù)闹?并且使該值為新的緊湊對齊值。如果你使用p o p 參量且內(nèi)部編譯器堆棧是空的,則緊湊值為命令行給定的值, 并且將產(chǎn)生一個警告信息。若你使用p o p 且指定一
個n 的值, 該值將成為新的緊湊值。若你使用p o p 且指定一個標(biāo)識符,  所有存儲在堆棧中的值將從棧中刪除, 直到找到一個匹配的標(biāo)識符, 這個與標(biāo)識符相關(guān)的緊湊值也從棧中移出, 并且這個僅在標(biāo)識符入棧之前存在的緊湊值成為新的緊湊值。如果未找到匹配的標(biāo)識符, 將使用命令行設(shè)置的緊湊值, 并且將產(chǎn)生一個一級警告。缺省緊湊對齊為8 。p a c k 編譯指示的新的增強(qiáng)功能讓你編寫頭文件, 確保在遇到該頭文件的前后的緊湊值是一樣的。

    什么是內(nèi)存對齊

    考慮下面的結(jié)構(gòu):

         struct foo
         {
           char c1;
           short s;
           char c2;
           int i;
          };
   
    假設(shè)這個結(jié)構(gòu)的成員在內(nèi)存中是緊湊排列的,假設(shè)c1的地址是0,那么s的地址就應(yīng)該是1,c2的地址就是3,i的地址就是4。也就是
    c1 00000000, s 00000001, c2 00000003, i 00000004。

    可是,我們在Visual c/c++ 6中寫一個簡單的程序:

         struct foo a;
    printf("c1 %p, s %p, c2 %p, i %p\n",
        (unsigned int)(void*)&a.c1 - (unsigned int)(void*)&a,
        (unsigned int)(void*)&a.s - (unsigned int)(void*)&a,
        (unsigned int)(void*)&a.c2 - (unsigned int)(void*)&a,
        (unsigned int)(void*)&a.i - (unsigned int)(void*)&a);
    運(yùn)行,輸出:
         c1 00000000, s 00000002, c2 00000004, i 00000008。

    為什么會這樣?這就是內(nèi)存對齊而導(dǎo)致的問題。

為什么會有內(nèi)存對齊

    以下內(nèi)容節(jié)選自《Intel Architecture 32 Manual》。
    字,雙字,和四字在自然邊界上不需要在內(nèi)存中對齊。(對字,雙字,和四字來說,自然邊界分別是偶數(shù)地址,可以被4整除的地址,和可以被8整除的地址。)
    無論如何,為了提高程序的性能,數(shù)據(jù)結(jié)構(gòu)(尤其是棧)應(yīng)該盡可能地在自然邊界上對齊。原因在于,為了訪問未對齊的內(nèi)存,處理器需要作兩次內(nèi)存訪問;然而,對齊的內(nèi)存訪問僅需要一次訪問。
    一個字或雙字操作數(shù)跨越了4字節(jié)邊界,或者一個四字操作數(shù)跨越了8字節(jié)邊界,被認(rèn)為是未對齊的,從而需要兩次總線周期來訪問內(nèi)存。一個字起始地址是奇數(shù)但卻沒有跨越字邊界被認(rèn)為是對齊的,能夠在一個總線周期中被訪問。
    某些操作雙四字的指令需要內(nèi)存操作數(shù)在自然邊界上對齊。如果操作數(shù)沒有對齊,這些指令將會產(chǎn)生一個通用保護(hù)異常(#GP)。雙四字的自然邊界是能夠被16 整除的地址。其他的操作雙四字的指令允許未對齊的訪問(不會產(chǎn)生通用保護(hù)異常),然而,需要額外的內(nèi)存總線周期來訪問內(nèi)存中未對齊的數(shù)據(jù)。

編譯器對內(nèi)存對齊的處理

    缺省情況下,c/c++編譯器默認(rèn)將結(jié)構(gòu)、棧中的成員數(shù)據(jù)進(jìn)行內(nèi)存對齊。因此,上面的程序輸出就變成了:
c1 00000000, s 00000002, c2 00000004, i 00000008。
編譯器將未對齊的成員向后移,將每一個都成員對齊到自然邊界上,從而也導(dǎo)致了整個結(jié)構(gòu)的尺寸變大。盡管會犧牲一點空間(成員之間有空洞),但提高了性能。
也正是這個原因,我們不可以斷言sizeof(foo) == 8。在這個例子中,sizeof(foo) == 12。

如何避免內(nèi)存對齊的影響

    那么,能不能既達(dá)到提高性能的目的,又能節(jié)約一點空間呢?有一點小技巧可以使用。比如我們可以將上面的結(jié)構(gòu)改成:

    struct bar
    {
        char c1;
        char c2;
        short s;
        int i;
    };
    這樣一來,每個成員都對齊在其自然邊界上,從而避免了編譯器自動對齊。在這個例子中,sizeof(bar) == 8。

    這個技巧有一個重要的作用,尤其是這個結(jié)構(gòu)作為API的一部分提供給第三方開發(fā)使用的時候。第三方開發(fā)者可能將編譯器的默認(rèn)對齊選項改變,從而造成這個結(jié)構(gòu)在你的發(fā)行的DLL中使用某種對齊方式,而在第三方開發(fā)者哪里卻使用另外一種對齊方式。這將會導(dǎo)致重大問題。
    比如,foo結(jié)構(gòu),我們的DLL使用默認(rèn)對齊選項,對齊為
c1 00000000, s 00000002, c2 00000004, i 00000008,同時sizeof(foo) == 12。
而第三方將對齊選項關(guān)閉,導(dǎo)致
    c1 00000000, s 00000001, c2 00000003, i 00000004,同時sizeof(foo) == 8。

如何使用c/c++中的對齊選項

    vc6中的編譯選項有 /Zp[1|2|4|8|16] ,/Zp1表示以1字節(jié)邊界對齊,相應(yīng)的,/Zpn表示以n字節(jié)邊界對齊。n字節(jié)邊界對齊的意思是說,一個成員的地址必須安排在成員的尺寸的整數(shù)倍地址上或者是n的整數(shù)倍地址上,取它們中的最小值。也就是:
    min ( sizeof ( member ),  n)
    實際上,1字節(jié)邊界對齊也就表示了結(jié)構(gòu)成員之間沒有空洞。
    /Zpn選項是應(yīng)用于整個工程的,影響所有的參與編譯的結(jié)構(gòu)。
    要使用這個選項,可以在vc6中打開工程屬性頁,c/c++頁,選擇Code Generation分類,在Struct member alignment可以選擇。

    要專門針對某些結(jié)構(gòu)定義使用對齊選項,可以使用#pragma pack編譯指令。指令語法如下:
#pragma pack( [ show ] | [ push | pop ] [, identifier ] , n  )
    意義和/Zpn選項相同。比如:

    #pragma pack(1)
    struct foo_pack
    {
        char c1;
        short s;
        char c2;
        int i;
    };
    #pragma pack()

    棧內(nèi)存對齊

    我們可以觀察到,在vc6中棧的對齊方式不受結(jié)構(gòu)成員對齊選項的影響。(本來就是兩碼事)。它總是保持對齊,而且對齊在4字節(jié)邊界上。

    驗證代碼

    #include <stdio.h>

    struct foo
    {
        char c1;
        short s;
        char c2;
        int i;
    };

    struct bar
    {
        char c1;
        char c2;
        short s;
        int i;
    };

    #pragma pack(1)
    struct foo_pack
    {
        char c1;
        short s;
        char c2;
        int i;
    };
    #pragma pack()


    int main(int argc, char* argv[])
    {
        char c1;
        short s;
        char c2;
        int i;

    struct foo a;
    struct bar b;
    struct foo_pack p;

    printf("stack c1 %p, s %p, c2 %p, i %p\n",
        (unsigned int)(void*)&c1 - (unsigned int)(void*)&i,
        (unsigned int)(void*)&s - (unsigned int)(void*)&i,
        (unsigned int)(void*)&c2 - (unsigned int)(void*)&i,
        (unsigned int)(void*)&i - (unsigned int)(void*)&i);

    printf("struct foo c1 %p, s %p, c2 %p, i %p\n",
        (unsigned int)(void*)&a.c1 - (unsigned int)(void*)&a,
        (unsigned int)(void*)&a.s - (unsigned int)(void*)&a,
        (unsigned int)(void*)&a.c2 - (unsigned int)(void*)&a,
        (unsigned int)(void*)&a.i - (unsigned int)(void*)&a);

    printf("struct bar c1 %p, c2 %p, s %p, i %p\n",
        (unsigned int)(void*)&b.c1 - (unsigned int)(void*)&b,
        (unsigned int)(void*)&b.c2 - (unsigned int)(void*)&b,
        (unsigned int)(void*)&b.s - (unsigned int)(void*)&b,
        (unsigned int)(void*)&b.i - (unsigned int)(void*)&b);

    printf("struct foo_pack c1 %p, s %p, c2 %p, i %p\n",
        (unsigned int)(void*)&p.c1 - (unsigned int)(void*)&p,
        (unsigned int)(void*)&p.s - (unsigned int)(void*)&p,
        (unsigned int)(void*)&p.c2 - (unsigned int)(void*)&p,
        (unsigned int)(void*)&p.i - (unsigned int)(void*)&p);

    printf("sizeof foo is %d\n", sizeof(foo));
    printf("sizeof bar is %d\n", sizeof(bar));
    printf("sizeof foo_pack is %d\n", sizeof(foo_pack));
   
    return 0;


    }




posted on 2007-12-23 20:03 sdfasdf 閱讀(17422) 評論(1)  編輯 收藏 引用 所屬分類: C++

Feedback

# re: 關(guān)于pragma pack的用法(四) C++中的內(nèi)存對齊問題 2008-01-03 20:01 guojianlee
不同編譯器的,的內(nèi)存對齊的參數(shù)選項不同,要注意。上次因為這個問題。搞了半死  回復(fù)  更多評論
  

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            国产日韩精品在线观看| 亚洲激情视频在线| 国内伊人久久久久久网站视频| 欧美日本精品一区二区三区| 免费人成精品欧美精品| 免费成人高清在线视频| 免费视频一区| 欧美日韩精品二区| 国产精品久久久久久影视| 国产精品二区二区三区| 国产一区美女| 亚洲另类自拍| 久久国内精品视频| 欧美91精品| 99国产精品久久久久久久| 新片速递亚洲合集欧美合集| 蜜臀av在线播放一区二区三区| 欧美噜噜久久久xxx| 国产精品久久久久久影视 | 亚洲欧美综合网| 久久精品国产亚洲一区二区| 模特精品在线| 亚洲午夜激情网页| 久久午夜电影| 国产精品久久久久久久久搜平片| 韩国三级电影久久久久久| 亚洲黄色av| 欧美一区亚洲| aⅴ色国产欧美| 久久人人爽人人爽爽久久| 欧美日韩一区综合| 亚洲国产成人tv| 久久狠狠婷婷| 亚洲最新在线视频| 嫩草伊人久久精品少妇av杨幂| 欧美性猛交xxxx免费看久久久| 一区二区三区在线观看欧美| 亚洲一卡二卡三卡四卡五卡| 久热爱精品视频线路一| 亚洲一区视频| 欧美日韩岛国| 亚洲日本成人网| 久久综合伊人77777麻豆| 亚洲女同精品视频| 欧美日韩精品久久久| 一区二区三区在线观看视频| 欧美在线日韩| 亚洲在线观看免费| 国产精品久久久999| 亚洲最新在线视频| 最新成人在线| 欧美激情成人在线| 最新国产成人av网站网址麻豆| 久久久久久久久久久久久女国产乱 | 久久精品91| 国产视频在线一区二区 | 久久久天天操| 性感少妇一区| 国产中文一区二区| 久久久久久久久久久久久9999 | 欧美大片专区| 久久久久久久久一区二区| 国产女人水真多18毛片18精品视频| 亚洲一二三四久久| 亚洲午夜电影网| 国产麻豆9l精品三级站| 欧美一级在线播放| 午夜精品亚洲| 国产一区二区精品在线观看| 久久久久9999亚洲精品| 久久精品成人一区二区三区蜜臀| 国产一区视频网站| 久久亚洲综合色| 久久一区二区三区四区| 亚洲日本电影| 一区二区三区欧美视频| 欧美少妇一区二区| 欧美在线国产精品| 久久九九热re6这里有精品 | 91久久精品国产91久久性色| 亚洲福利视频在线| 欧美男人的天堂| 香蕉av福利精品导航| 欧美一区二区三区播放老司机| 国产日产精品一区二区三区四区的观看方式| 亚洲一区在线免费| 久久久久久久网站| 99在线视频精品| 午夜国产一区| 亚洲国产一区二区a毛片| 亚洲国产高清视频| 欧美日韩专区| 欧美 日韩 国产在线| 欧美精品入口| 久热综合在线亚洲精品| 国产精品都在这里| 欧美成人国产一区二区| 国产精品高清在线| 女人色偷偷aa久久天堂| 国产精品爽爽爽| 欧美激情一区二区三区在线视频 | 国产一区二区中文字幕免费看| 蘑菇福利视频一区播放| 欧美日韩在线一区二区三区| 久久久久久电影| 亚洲欧洲综合另类在线| 欧美激情一区二区久久久| 欧美日韩岛国| 欧美电影资源| 国内久久婷婷综合| 亚洲特级片在线| 亚洲理伦电影| 久久久久久久久伊人| 欧美一级专区免费大片| 欧美日韩在线电影| 亚洲福利免费| 伊人久久大香线蕉综合热线 | 亚洲日本中文| 在线免费观看成人网| 性欧美精品高清| 亚洲一区二区三区午夜| 欧美激情一区二区三区四区| 蜜桃精品一区二区三区| 国产亚洲综合精品| 亚洲女人天堂av| 亚洲一级高清| 欧美视频一区二区在线观看 | 久久国产精品黑丝| 国产精品久久久久久久久搜平片 | 国产一区二区黄色| 亚洲一区二区精品视频| 亚洲视频免费看| 欧美日韩免费区域视频在线观看| 欧美激情精品久久久久久免费印度 | 91久久极品少妇xxxxⅹ软件| 亚洲高清中文字幕| 久久亚洲综合色| 欧美电影电视剧在线观看| 一色屋精品视频免费看| 久久久精品国产免费观看同学| 久久久久久亚洲综合影院红桃| 国产视频亚洲| 久久久久久亚洲精品中文字幕| 久久人人97超碰国产公开结果| 国产日韩欧美综合精品| 欧美一级视频精品观看| 久久伊伊香蕉| 伊人成人网在线看| 欧美成人精精品一区二区频| 亚洲黄色三级| 亚洲桃花岛网站| 国产精品嫩草99a| 欧美在线观看一区二区| 欧美福利专区| 正在播放日韩| 国产精品一区二区在线观看| 亚洲一区二区伦理| 欧美婷婷久久| 亚洲国产精品999| 曰韩精品一区二区| 欧美福利影院| 亚洲一区在线视频| 噜噜噜躁狠狠躁狠狠精品视频| 亚洲激情第一页| 国产精品久久久久久久久免费| 亚洲欧洲99久久| 亚洲国产欧美国产综合一区| 亚洲深爱激情| 一区二区在线免费观看| 欧美精品一区二区视频| 亚洲一区亚洲二区| 欧美成人免费大片| 一区二区国产日产| 国产一区二区三区四区hd| 欧美国产一区在线| 午夜精品久久| 亚洲激情国产| 葵司免费一区二区三区四区五区| 日韩视频在线观看免费| 国产一区二区精品久久91| 欧美片在线播放| 久久av一区| 中文在线资源观看网站视频免费不卡 | 一本久道久久综合中文字幕| 国产美女精品视频| 免费不卡中文字幕视频| 小辣椒精品导航| 亚洲人午夜精品免费| 久久青青草原一区二区| 亚洲综合色激情五月| 最新国产乱人伦偷精品免费网站| 国产精品美腿一区在线看| 欧美激情成人在线| 久久久久成人精品| 亚洲欧美一区二区三区在线| 亚洲精品久久视频| 欧美r片在线| 美女精品在线观看| 久久久www成人免费精品| 亚洲欧美日韩系列|