#pragma pack(push,1) #pragma pack(pop)
1 引子
????? 在程序中,有的時(shí)候我們定義結(jié)構(gòu)體的時(shí)候,要用#pragma pack(push,1) & #pragma pack(pop)類似代碼將結(jié)構(gòu)體包起來(lái)。
一般形式如下:
#pragma pack(push,1);
struct A
{
?
} ;
#pragma pack(pop);
這么做有什么目的呢?
?注:下列內(nèi)容來(lái)自網(wǎng)絡(luò)。
2 #pragma pack簡(jiǎn)介
#pragma pack是指定數(shù)據(jù)在內(nèi)存中的對(duì)齊方式,
在C語(yǔ)言中,結(jié)構(gòu)是一種復(fù)合數(shù)據(jù)類型,其構(gòu)成元素既可以是基本數(shù)據(jù)類型(如int、long、float等)的變量,也可以是一些復(fù)合數(shù)據(jù)類型(如數(shù)組、結(jié)構(gòu)、聯(lián)合等)的數(shù)據(jù)單元。在結(jié)構(gòu)中,編譯器為結(jié)構(gòu)的每個(gè)成員按其自然對(duì)界(alignment)條件分配空間。各個(gè)成員按照它們被聲明的順序在內(nèi)存中順序存儲(chǔ),第一個(gè)成員的地址和整個(gè)結(jié)構(gòu)的地址相同。
例 1:
struct sample
{
char a;
double b;
};
若不用#pragma pack(1)和#pragma pack()括起來(lái),則sample按編譯器默認(rèn)方式對(duì)齊(成員中size最大的那個(gè))。即按8字節(jié)(double)對(duì)齊,則sizeof(sample)==16.成員char a占了8個(gè)字節(jié)(其中7個(gè)是空字節(jié))
若用#pragma pack(1),則sample按1字節(jié)方式對(duì)齊sizeof(sample)==9.(無(wú)空字節(jié))
例 2:下面的結(jié)構(gòu)各成員空間分配情況:
struct test
{
???? char x1;
???? short x2;
???? float x3;
???? char x4;
};
???? 結(jié)構(gòu)的第一個(gè)成員x1,其偏移地址為0,占據(jù)了第1個(gè)字節(jié)。第二個(gè)成員x2為short類型,其起始地址必須2字節(jié)對(duì)界,因此,編譯器在x2和x1之間填充了一個(gè)空字節(jié)。結(jié)構(gòu)的第三個(gè)成員x3和第四個(gè)成員x4恰好落在其自然對(duì)界地址上,在它們前面不需要額外的填充字節(jié)。在test結(jié)構(gòu)中,成員x3要求4字節(jié)對(duì)界,是該結(jié)構(gòu)所有成員中要求的最大對(duì)界單元,因而test結(jié)構(gòu)的自然對(duì)界條件為4字節(jié),編譯器在成員x4后面填充了3個(gè)空字節(jié)。整個(gè)結(jié)構(gòu)所占據(jù)空間為12字節(jié)。更改C編譯器的缺省字節(jié)對(duì)齊方式
???? 在缺省情況下,C編譯器為每一個(gè)變量或是數(shù)據(jù)單元按其自然對(duì)界條件分配空間。一般地,可以通過(guò)下面的方法來(lái)改變?nèi)笔〉膶?duì)界條件:
· 使用偽指令#pragma pack (n),C編譯器將按照n個(gè)字節(jié)對(duì)齊。
???? · 使用偽指令#pragma pack (),取消自定義字節(jié)對(duì)齊方式。
???? 另外,還有如下的一種方式:
???? · __attribute((aligned (n))),讓所作用的結(jié)構(gòu)成員對(duì)齊在n字節(jié)自然邊界上。如果結(jié)構(gòu)中有成員的長(zhǎng)度大于n,則按照最大成員的長(zhǎng)度來(lái)對(duì)齊。
???? · __attribute__ ((packed)),取消結(jié)構(gòu)在編譯過(guò)程中的優(yōu)化對(duì)齊,按照實(shí)際占用字節(jié)數(shù)進(jìn)行對(duì)齊。
以上的n = 1, 2, 4, 8, 16... 第一種方式較為常見(jiàn)。
3 應(yīng)用實(shí)例
在網(wǎng)絡(luò)協(xié)議編程中,經(jīng)常會(huì)處理不同協(xié)議的數(shù)據(jù)報(bào)文。一種方法是通過(guò)指針偏移的方法來(lái)得到各種信息,但這樣做不僅編程復(fù)雜,而且一旦協(xié)議有變化,程序修改起來(lái)也比較麻煩。在了解了編譯器對(duì)結(jié)構(gòu)空間的分配原則之后,我們完全可以利用這一特性定義自己的協(xié)議結(jié)構(gòu),通過(guò)訪問(wèn)結(jié)構(gòu)的成員來(lái)獲取各種信息。這樣做,不僅簡(jiǎn)化了編程,而且即使協(xié)議發(fā)生變化,我們也只需修改協(xié)議結(jié)構(gòu)的定義即可,其它程序無(wú)需修改,省時(shí)省力。下面以TCP協(xié)議首部為例,說(shuō)明如何定義協(xié)議結(jié)構(gòu)。其協(xié)議結(jié)構(gòu)定義如下:
?
?
| #pragma pack(1) // 按照1字節(jié)方式進(jìn)行對(duì)齊 |
總結(jié)
以上是生活随笔為你收集整理的#pragma pack(push,1) #pragma pack(pop)的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。