• <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>
            posts - 311, comments - 0, trackbacks - 0, articles - 0
              C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理
             首先請大家先看下面代碼:
            typedef struct
            {
            UINT32 NumElements;
            union
            {
            UINT32 ObjectHandle;
            }Entry;
            }STR_ARRAY, *PSTR_ARRAY;

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

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

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

            什么是內存對齊

            考慮下面的結構:

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

            假設這個結構的成員在內存中是緊湊排列的,假設c1的地址是0,那么s的地址就應該是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);
            運行,輸出:
            c1 00000000, s 00000002, c2 00000004, i 00000008。

            為什么會這樣?這就是內存對齊而導致的問題。

            為什么會有內存對齊

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

            編譯器對內存對齊的處理

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

            如何避免內存對齊的影響

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

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

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

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

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

            要專門針對某些結構定義使用對齊選項,可以使用#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()

            棧內存對齊

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

            驗證代碼

            #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;


            }

            亚洲天堂久久久| 亚洲国产日韩综合久久精品| 久久精品国产亚洲AV无码麻豆| 国产精品对白刺激久久久| 99久久无码一区人妻| 精品国产乱码久久久久软件| 青草国产精品久久久久久| 精品99久久aaa一级毛片| 中文字幕乱码久久午夜| 久久精品成人免费网站| 久久精品国产清自在天天线| 精品多毛少妇人妻AV免费久久| 亚洲AV无码一区东京热久久| 久久久亚洲精品蜜桃臀| 99久久久国产精品免费无卡顿 | 久久国产一区二区| 热久久国产欧美一区二区精品 | 久久精品国产亚洲av日韩| 久久e热在这里只有国产中文精品99| 久久精品成人欧美大片| 人妻精品久久久久中文字幕| 人人狠狠综合久久亚洲88| 久久久精品2019免费观看| 思思久久精品在热线热| 久久久久久久综合日本| 99久久免费只有精品国产| 久久九九有精品国产23百花影院| 久久久国产精华液| 99久久做夜夜爱天天做精品| 国产精品欧美亚洲韩国日本久久| 国产精品久久久久天天影视| 亚洲∧v久久久无码精品| 亚洲精品无码成人片久久| 狠狠色狠狠色综合久久| 色欲av伊人久久大香线蕉影院| 一本色综合久久| 亚洲精品乱码久久久久久自慰 | 久久777国产线看观看精品| AV狠狠色丁香婷婷综合久久| 久久99精品久久只有精品| 国内精品久久久久影院一蜜桃|