C語言中結(jié)構(gòu)體、聯(lián)合體的成員內(nèi)存對齊情況
前言
最近項(xiàng)目進(jìn)行中,遇到一個(gè)小問題,在數(shù)據(jù)協(xié)議傳輸過程中,我為了方便解析,就定義了一個(gè)結(jié)構(gòu)體,在數(shù)據(jù)的指針傳入函數(shù)的時(shí)候,我用定義好的結(jié)構(gòu)體進(jìn)行強(qiáng)制轉(zhuǎn)化,沒想到一直解析失敗,調(diào)試很久,終于反應(yīng)過來,在用結(jié)構(gòu)體指針對數(shù)據(jù)強(qiáng)制轉(zhuǎn)換時(shí),定義結(jié)構(gòu)體我沒有注意到數(shù)據(jù)對齊,因?yàn)樵诘讓訉?shí)現(xiàn)中,我傳入的數(shù)據(jù)buffer是排列整齊的,而強(qiáng)制轉(zhuǎn)化的結(jié)構(gòu)體格式中,我定義的時(shí)候沒有使用__attribute__((__packed__))或者_(dá)_packed強(qiáng)制數(shù)據(jù)對齊,導(dǎo)致結(jié)構(gòu)體成員真實(shí)排列會按照成員中最大的變量的格式進(jìn)行對其,缺少的地方被虛擬補(bǔ)充位置。
下面就稍微簡單描述一下結(jié)構(gòu)體數(shù)據(jù)對齊的講解:
圖片描述的兩種實(shí)現(xiàn)結(jié)構(gòu)對齊的聲明,適用于結(jié)構(gòu)體和聯(lián)合的聲明。
接下來展示幾組聲明結(jié)構(gòu)體后成員變量對齊的方式:
/*第一個(gè)示例*/ struct stc { char one; short two; char three; int four; } c,d; int main (void) { c.one=1; return 0; }
第一個(gè)示例代碼配合下方內(nèi)存排列的圖片,可以看到,在正常無特殊聲明的情況下,結(jié)構(gòu)體在內(nèi)存排列是按照結(jié)構(gòu)體成員中最大的變量的大小進(jìn)行排列的。
第一處示例代碼中,最大的成員變量是int型,一個(gè)int型在我使用的32位ARM環(huán)境中占4個(gè)byte,所以在排列中,最小的排列單位是4byte,而其他類型,char占1個(gè)byte,short占2個(gè)byte,在排列的第一行的4個(gè)byte中,一個(gè)char+一個(gè)short類型為3byte,所以需要補(bǔ)上1byte的虛擬空間,第二行的4byte中,還剩下一個(gè)char和int,int單獨(dú)占一行,所以char需要補(bǔ)上3byte才能排列整齊。
/*第二個(gè)示例*/ struct __attribute__((packed)) stc { char one; short two; char three; int four; } c,d; int main (void) { c.one=1; return 0; }
第二個(gè)示例代碼配合下方內(nèi)存排列的圖片,可以看到,代碼使用了__attribute__((packed))聲明,這個(gè)聲明的含義是,令相關(guān)的結(jié)構(gòu)體與聯(lián)合體強(qiáng)制一字節(jié)對齊。所以在內(nèi)存中排列中,按照1byte的數(shù)據(jù)對齊方式,成員變量緊密排布。
/*第三個(gè)示例*/ #pragma pack (2) struct stc { char one; short two; char three; int four; } c,d; int main (void) { c.one=1; return 0; }
/*第四個(gè)示例*/ #pragma pack (4) struct stc { char one; short two; char three; int four; } c,d; int main (void) { c.one=1; return 0; }
第三、四個(gè)示例代碼配合下方內(nèi)存排列的圖片,可以看到,代碼使用了#pragma pack (n)聲明,這個(gè)聲明的含義是,令相關(guān)的結(jié)構(gòu)體與聯(lián)合體強(qiáng)制N字節(jié)對齊,這個(gè)聲明和__attribute__((packed))功能類似,但是__attribute__((packed))只能進(jìn)行一字節(jié)強(qiáng)制對齊,而#pragma pack (n)對齊字節(jié)數(shù),由n進(jìn)行控制,所以有很多的靈活性。具體使用可以從下圖成員對齊情況了解,此處就不進(jìn)行贅述了。
總結(jié)
到此這篇關(guān)于C語言中結(jié)構(gòu)體、聯(lián)合體的成員內(nèi)存對齊情況的文章就介紹到這了,更多相關(guān)C語言結(jié)構(gòu)體、聯(lián)合體內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- C語言中結(jié)構(gòu)體的內(nèi)存對齊規(guī)則講解
- C語言詳解結(jié)構(gòu)體的內(nèi)存對齊與大小計(jì)算
- C語言詳解熱門考點(diǎn)結(jié)構(gòu)體內(nèi)存對齊
- C語言 詳細(xì)分析結(jié)構(gòu)體的內(nèi)存對齊
- C語言結(jié)構(gòu)體中內(nèi)存對齊的問題理解
- C語言結(jié)構(gòu)體內(nèi)存對齊詳解
- C語言熱門考點(diǎn)結(jié)構(gòu)體與內(nèi)存對齊詳解
- C語言中結(jié)構(gòu)體與內(nèi)存對齊實(shí)例解析
- C語言詳細(xì)分析結(jié)構(gòu)體的內(nèi)存對齊規(guī)則
相關(guān)文章
C++ 中const對象與const成員函數(shù)的實(shí)例詳解
這篇文章主要介紹了C++ 中const對象與const成員函數(shù)的實(shí)例詳解的相關(guān)資料,希望通過本文能讓大家徹底掌握該如何使用,需要的朋友可以參考下2017-08-08如何使用C++結(jié)合OpenCV進(jìn)行圖像處理與分類
在計(jì)算機(jī)視覺領(lǐng)域,OpenCV與C++結(jié)合能高效處理和分類圖像,C++的高執(zhí)行效率適合大規(guī)模數(shù)據(jù)處理,OpenCV提供豐富的功能,如圖像預(yù)處理和機(jī)器學(xué)習(xí)算法,安裝OpenCV需要配置環(huán)境和添加庫文件,本文詳細(xì)介紹了使用C++和OpenCV進(jìn)行圖像分類的過程,包括使用SVM和深度學(xué)習(xí)模型2024-09-09C++實(shí)現(xiàn)LeetCode(228.總結(jié)區(qū)間)
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(228.總結(jié)區(qū)間),本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07C語言實(shí)現(xiàn)Fibonacci數(shù)列遞歸
這篇文章主要介紹了C語言實(shí)現(xiàn)Fibonacci數(shù)列遞歸,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-02-02C C++算法題解LeetCode1408數(shù)組中的字符串匹配
這篇文章主要為大家介紹了C C++算法題解LeetCode1408數(shù)組中的字符串匹配示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-10-10C++中指針的數(shù)據(jù)類型和運(yùn)算相關(guān)知識小結(jié)
這篇文章主要介紹了C++中指針的數(shù)據(jù)類型和運(yùn)算相關(guān)知識小結(jié),是C++入門學(xué)習(xí)中的基礎(chǔ)知識,需要的朋友可以參考下2015-09-09Visual Studio 2022最新版安裝教程(圖文詳解)
本文主要介紹了Visual Studio 2022最新版安裝教程,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01C++深入探索類和對象之封裝及class與struct的區(qū)別
C++?類與對象涉及的知識點(diǎn)非常廣泛,所以我準(zhǔn)備寫成幾個(gè)特定的部分來作為博文分享,這次的blog將詳細(xì)講解類的屬性、行為、訪問權(quán)限,class與struct的區(qū)別以及具體案例,希望能夠?qū)δ銈冇袔椭?,解決入門小白或者對這方面了解不多的朋友們,那么接下來開始今天的內(nèi)容2022-05-05