資料在記憶體中對齊方式 pragma pack

2021-07-25 15:27:34 字數 2039 閱讀 5455

在程式中,有的時候我們定義結構體的時候,要用#pragma pack(push,1) & #pragma pack(pop)類似**將結構體包起來。

一般形式如下:

#pragma pack(push,1);

struct a ;

#pragma pack(pop);

這麼做有什麼目的呢?

注:下列內容來自網路。

2 #pragma pack簡介

#pragma pack是指定資料在記憶體中的對齊方式,

在c語言中,結構是一種復合資料型別,其構成元素既可以是基本資料型別(如int、long、float等)的變數,也可以是一些復合資料型別(如陣列、結構、聯合等)的資料單元。在結構中,編譯器為結構的每個成員按其自然對界(alignment)條件分配空間。各個成員按照它們被宣告的順序在記憶體中順序儲存,第乙個成員的位址和整個結構的位址相同。

例 1:

struct sample 

; 若不用#pragma pack(1)和#pragma pack()括起來,則sample按編譯器預設方式對齊(成員中size最大的那個)。即按8位元組(double)對齊,則sizeof(sample)==16.成員char a佔了8個位元組(其中7個是空位元組) 

若用#pragma pack(1),則sample按1位元組方式對齊sizeof(sample)==9.(無空位元組)

例 2:下面的結構各成員空間分配情況:

struct test 

;結構的第乙個成員x1,其偏移位址為0,佔據了第1個位元組。第二個成員x2為short型別,其起始位址必須2位元組對界,因此,編譯器在x2和x1之間填充了乙個空位元組。結構的第三個成員x3和第四個成員x4恰好落在其自然對界位址上,在它們前面不需要額外的填充位元組。在test結構中,成員x3要求4位元組對界,是該結構所有成員中要求的最大對界單元,因而test結構的自然對界條件為4位元組,編譯器在成員x4後面填充了3個空位元組。整個結構所佔據空間為12位元組。更改c編譯器的預設位元組對齊方式

[cpp]view plain

copy

#include 

#pragma pack(push,1) //pragma pack(push)

//pragma pack(1)

typedef

struct

_a  

a;  

#pragma pack(pop)

intmain(

intargc,

char

*argv)    

在預設情況下,c編譯器為每乙個變數或是資料單元按其自然對界條件分配空間。一般地,可以通過下面的方法來改變預設的對界條件:

· 使用偽指令#pragma pack (n),c編譯器將按照n個位元組對齊。

· 使用偽指令#pragma pack (),取消自定義位元組對齊方式。

另外,還有如下的一種方式:

· __attribute((aligned (n))),讓所作用的結構成員對齊在n位元組自然邊界上。如果結構中有成員的長度大於n,則按照最大成員的長度來對齊。

· __attribute__ ((packed)),取消結構在編譯過程中的優化對齊,按照實際占用位元組數進行對齊。

以上的n = 1, 2, 4, 8, 16... 第一種方式較為常見。

3 應用例項

在網路協議程式設計中,經常會處理不同協議的資料報文。一種方法是通過指標偏移的方法來得到各種資訊,但這樣做不僅程式設計複雜,而且一旦協議有變化,程式修改起來也比較麻煩。在了解了編譯器對結構空間的分配原則之後,我們完全可以利用這一特性定義自己的協議結構,通過訪問結構的成員來獲取各種資訊。這樣做,不僅簡化了程式設計,而且即使協議發生變化,我們也只需修改協議結構的定義即可,其它程式無需修改,省時省力。下面以tcp協議首部為例,說明如何定義協議結構。其協議結構定義如下: 

#pragma pack(1) // 按照1位元組方式進行對齊

struct tcpheader 

; #pragma pack()

結構體在記憶體中對齊

剛剛完成乙個檔案的遷移程式,其中遇到了結構體對齊的問題,所以拿出來說說,與各位博友們分享。我的程式很簡單,就是把之前通過乙個結構體 fwrite 到檔案 a 裡的內容讀出,然後轉給另乙個結構體儲存。程式是簡單,但我擔心的是之前把結構體 fwrite 到檔案 a 的程式對齊結構體規則是怎樣的?一定要知...

資料在記憶體中的儲存方式

之前在面試的時候有遇到乙個面試題 記憶體中的顯示 輸入出其實這就是little endian 小端序列 的儲存形式 比方說我有乙個0xa5b1的乙個資料。如果當前的機器是小端序列那麼在 記憶體中高位位址 存放的就是a5 example 0x40000001 記憶體中低位位址 存放的就是b1 exam...

資料在記憶體中的儲存方式

例如,將十進位制178.125表示成機器內的32個位元組的二進位制形式.將128.125表示成二進位制數 178.125 十進位制數 10110010.001 二進位制形式 將二進位制形式的浮點實數轉化為規格化的形式 小數點向左移動7個二進位制位可以得到 10110010.001 1.0110010...