c語言結(jié)構(gòu)體字節(jié)對齊的實現(xiàn)方法
1.什么是字節(jié)對齊
在c語言的結(jié)構(gòu)體里面一般會按照某種規(guī)則去進行字節(jié)對齊。
我們先看一段代碼:
struct st1 { char name; double age; char sex; }; //32位下 sizeof(struct st1) = 16 //64位下 sizeof(struct st1) = 24 struct st2 { char a; char b; char c; }; //32位和64位下, sizeof(struct st2)都是3個字節(jié)
從以上結(jié)果可以看出,結(jié)構(gòu)體st1在32位下是按照4個字節(jié)來對齊的,在64位下則是按照8個字節(jié)來對齊的,結(jié)構(gòu)體st2則不管32位還是64位則都是按照1個字節(jié)對齊的。
那么我們可以總結(jié)出對齊規(guī)則如下:
- 在所有結(jié)構(gòu)體成員的字節(jié)長度都沒有超出操作系統(tǒng)基本字節(jié)單位(32位操作系統(tǒng)是4,64位操作系統(tǒng)是8)的情況下,按照結(jié)構(gòu)體中字節(jié)最大的變量長度來對齊;
- 若結(jié)構(gòu)體中某個變量字節(jié)超出操作系統(tǒng)基本字節(jié)單位,那么就按照系統(tǒng)字節(jié)單位來對齊。
注意:并不是32位就直接按照4個字節(jié)對齊,64位按照8個字節(jié)對齊。
2.為什么要有字節(jié)對齊
首先普及一點小知識,cpu一次能讀取多少內(nèi)存要看數(shù)據(jù)總線是多少位,如果是16位,則一次只能讀取2個字節(jié),如果是32位,則可以讀取4個字節(jié),并且cpu不能跨內(nèi)存區(qū)間訪問。
假設(shè)有這樣一個結(jié)構(gòu)體如下:
struct st3 { char a; int b; }; //那么根據(jù)我們第1節(jié)所說的規(guī)則,在32位系統(tǒng)下,它就應(yīng)該是8個字節(jié)的。
假設(shè)地址空間是類似下面這樣的:
在沒有字節(jié)對齊的情況下,變量a就是占用了0x00000001這一個字節(jié),而變量b則是占用了0x00000002~0x000000005這四個字節(jié),那么cpu如果想從內(nèi)存中讀取變量b,首先要從變量b的開始地址0x00000002讀到0x0000004,然后再讀取一次0x00000005這個字節(jié),相當于讀一個int,cpu從內(nèi)存讀取了兩次。
而如果進行字節(jié)對齊的話,變量a還是占用了0x00000001這一個字節(jié),而變量b則是占用了0x00000005~0x00000008這四個字節(jié),那么cpu要讀取變量b的話,就直接一次性從0x00000005讀到0x00000008,就一次全部讀取出來了。
所以說,字節(jié)對齊的根本原因其實在于cpu讀取內(nèi)存的效率問題,對齊以后,cpu讀取內(nèi)存的效率會更快。但是這里有個問題,就是對齊的時候0x00000002~0x00000004這三個字節(jié)是浪費的,所以字節(jié)對齊實際上也有那么點以空間換時間的意思,具體寫代碼的時候怎么選擇,其實是看個人的。
3.手動設(shè)置對齊
什么情況下需要手動設(shè)置對齊:
- 設(shè)計不同CPU下的通信協(xié)議,比如兩臺服務(wù)器之間進行網(wǎng)絡(luò)通信,共用一個結(jié)構(gòu)體時,需要手動設(shè)置對齊規(guī)則,確保兩邊結(jié)構(gòu)體長度一直;
- 編寫硬件驅(qū)動程序時寄存器的結(jié)構(gòu);
手動設(shè)置對齊方式有兩種:
代碼里添加預(yù)編譯標識:
//用法如下 #pragma pack(n)//表示它后面的代碼都按照n個字節(jié)對齊 struct st3 { char a; int b; }; #pragma pack()//取消按照n個字節(jié)對齊,是對#pragma pack(n)的一個反向操作 //這里計算sizeof(st3)=5
上面這兩行其實就類似于開車的時候,走到某一段路的時候,發(fā)現(xiàn)一個限速60公里的指示牌,過了那一段路以后,又會有解除限速60公里的指示牌。
定義結(jié)構(gòu)體時:
//用法如下 struct bbb { char a; int b; }__attribute__((packed));//直接按照實際占用字節(jié)來對齊,其實就是相當于按照1個字節(jié)對齊了 //這里計算sizeof(st3)=5
4.結(jié)構(gòu)體比較方法
可以使用內(nèi)存比較函數(shù)memcpy進行結(jié)構(gòu)體比較,但因為結(jié)構(gòu)體對齊可能會有填充位不一致的情況,此時需要注意:
- 設(shè)置為1個字節(jié)對齊,使它沒有空位;
- 事先對結(jié)構(gòu)體進行初始化;
memcpy(char *dest, const char* src, int len); //頭文件#include<string.h>
到此這篇關(guān)于c語言結(jié)構(gòu)體字節(jié)對齊的實現(xiàn)方法的文章就介紹到這了,更多相關(guān)c語言結(jié)構(gòu)體字節(jié)對齊內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
windows系統(tǒng)下C++調(diào)用matlab程序的方法詳解
這篇文章主要給大家介紹了關(guān)于在windows系統(tǒng)下C++調(diào)用matlab程序的方法,文中通過示例代碼介紹的非常詳細,對大家學(xué)習(xí)或者使用C++具有一定的參考學(xué)習(xí)價值,需要的朋友們下面跟著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2017-08-08C語言中typedef的用法以及#define區(qū)別詳解
這篇文章主要給大家介紹了關(guān)于C語言中typedef用法以及#define區(qū)別的相關(guān)資料,typedef 是用來定義一種類型的新別名的,它不同于宏(#define),不是簡單的字符串替換。而#define只是簡單的字符串替換(原地擴展),需要的朋友可以參考下2021-07-07C++面向?qū)ο笳Z言自制多級菜單功能實現(xiàn)代碼
菜單類主要負責(zé)菜單的創(chuàng)建、修改、刪除,是包含菜單結(jié)構(gòu)組織和響應(yīng)函數(shù)的模型,用戶擁有充分的自主性,可根據(jù)需要自定義菜單顯示和響應(yīng)函數(shù),這篇文章主要介紹了C++面向?qū)ο笳Z言自制多級菜單,需要的朋友可以參考下2024-06-06C++實現(xiàn)學(xué)生信息管理系統(tǒng)(完整版)
這篇文章主要為大家詳細介紹了C++實現(xiàn)學(xué)生信息管理系統(tǒng),文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-06-06