程式中#pragma pack(1)的用途,但不知道它所代表的意義呢?以下將進行解說
#pragma pack(1)
作用:調整結構體的邊界對齊,讓其以1個位元組對齊。
範例一
01 |
typedef struct _sampleA{ |
02 |
char a; |
03 |
double b; |
04 |
}; |
05 |
06 |
void main() |
07 |
{ |
08 |
int len = sizeof (sampleA); |
09 |
printf ( "sizeof(sampleA)=%d\n" , len); // len = 16 |
10 |
} |
解說:
若不用#pragma pack(1)和#pragma pack()括起來,則sample按編譯器預設方式對齊(成員中size最大的那個)。即按8位元組(double)對齊,則sizeof(sampleA)=16.成員char a占了8個位元組(其中7個是空位元組)
範例二
01 |
#pragma pack(push) |
02 |
#pragma pack(1) |
03 |
typedef struct _sampleB |
04 |
{ |
05 |
char a; |
06 |
double b; |
07 |
}; |
08 |
#pragma pack(pop) |
09 |
10 |
void main() |
11 |
{ |
12 |
int len = sizeof (sampleB); |
13 |
printf ( "sizeof(sampleB)=%d\n" , len); // len = 9 |
14 |
} |
注意:
push 在改變為1之前先儲存原來的設定。
pop 恢復原來的設定。
#pragma pack(push)與#pragma pack(1)可合併為#pragma pack(push,1)。
解說:
若用#pragma pack(1),則sample按1位元組方式對齊sizeof(sampleB)=9,(空位元組),比較節省空間啦,有些場和還可使結構體更易於控制。
"#pragma pack(n)"是要Compiler以n位元組對齊。
"#pragma pack()"取消位元組對齊。
struct的定義區段,用"#pragma pack(1)"和"#pragma pack( )"包起來,就可以了使struct正確被存取。
我們再舉幾個例子來了解實際的情況
範例一
1 |
typedef struct { |
2 |
ULONG Data1; |
3 |
USHORT Data2; |
4 |
USHORT Data3; |
5 |
char Data4[8]; |
6 |
} EFI_GUID; |
7 |
8 |
EFI_GUID g; |
9 |
printf ( "%d\n" , sizeof (g)) |
範例二
01 |
#pragma pack(1) |
02 |
typedef struct { |
03 |
ULONG Data1; |
04 |
USHORT Data2; |
05 |
USHORT Data3; |
06 |
char Data4[8]; |
07 |
} EFI_GUID; |
08 |
#pragma pack() |
09 |
10 |
EFI_GUID g; |
11 |
printf ( "%d\n" , sizeof (g)) |
範例三
1 |
typedef struct { |
2 |
ULONG Data1; |
3 |
USHORT Data2; |
4 |
USHORT Data3; |
5 |
char Data4[9]; |
6 |
} EFI_GUID; |
7 |
8 |
EFI_GUID g; |
9 |
printf ( "%d\n" , sizeof (g)) |
範例四
01 |
#pragma pack(1) |
02 |
typedef struct { |
03 |
ULONG Data1; |
04 |
USHORT Data2; |
05 |
USHORT Data3; |
06 |
char Data4[9]; |
07 |
} EFI_GUID; |
08 |
#pragma pack() |
09 |
10 |
EFI_GUID g; |
11 |
printf ( "%d\n" , sizeof (g)) |
範例一,沒加#pragma pack(1),在32bit機器上將以4 byte為預設長度,結構方面就以最長byte的變數4 byte為主,剛好與預設長度一樣,所以Length = 4 + 2 + 2 + 8 = 16,16為4的倍數
範例二,有加#pragma pack(1),在結構方面就以各變數長度為主,所以Length = 4 + 2 + 2 + 8 = 16
範例三,沒加#pragma pack(1),在32bit機器上將以4byte為預設長度,在結構方面就以最長byte的變數4 byte為主,剛好與預設長度一樣,所以Length = 4 + 2 + 2 + 9 = 17,17不為4的倍數 ,須補足到4的倍數,所以擴增長度補齊到20
範例四,有加#pragma pack(1),在結構方面就以各變數長度為主,所以Length = 4 + 2 + 2 + 8 = 17