0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

什么是結(jié)構(gòu)體的字節(jié)對齊現(xiàn)象

麥辣雞腿堡 ? 來源:軒哥談芯 ? 作者:義軒 ? 2023-11-20 15:55 ? 次閱讀

什么是結(jié)構(gòu)體的字節(jié)對齊現(xiàn)象

程序員,咱都用代碼說話,先上 code:

(說明:以下代碼均在 ARM 平臺上,使用 Keil 進行編譯測試)

#define offset_of(TYPE, MEMBER) ((size_t) &((TYPE *)0)-MEMBER) //上面這個宏定義主要用于顯示結(jié)構(gòu)體成員變量相對結(jié)構(gòu)體起始地址的偏移
typedef structstu1{ int a; char b; int c;}stu1;
void main(){ LOG_INFO("rnrn====== Struct Test ======rnrn"); LOG_INFO("offset_of(stu1,a):t%dn",offset_of(stu1,a)); LOG_INFO("offset_of(stu1,b):t%dn",offset_of(stu1,b)); LOG_INFO("offset_of(stu1,c):t%dn",offset_of(stu1,c)); LOG_INFO("sizeof(stu1) :t%dn",sizeof(stu1)); return ;}

圖片

對于上面的運行結(jié)果,對字節(jié)對齊不了解的同學(xué)可能會疑惑,c的偏移量怎么會是8呢?不應(yīng)該是 5 嗎?

結(jié)構(gòu)體的大小怎么會是12呢?不應(yīng)該是 9 嗎?

不了解的同學(xué)可能會這樣理解:

c的偏移量是sizeof(int)+sizeof(char) = 5

結(jié)構(gòu)體stu1占用的內(nèi)存大小應(yīng)該是sizeof(int)+sizeof(char)+sizeof(int)=9。

通過下圖所示的stu1的內(nèi)存結(jié)構(gòu)可以知道,編譯器對變量存儲進行了一個特殊處理。

圖片

為了提高CPU的存儲速度,編譯器對一些變量的起始地址做了對齊處理。

在默認情況下,編譯器規(guī)定各成員變量存放的起始地址相對于結(jié)構(gòu)體的起始地址的偏移量,必須為該變量的類型所占用的字節(jié)數(shù)和編譯器編譯過程中采用的字節(jié)對齊數(shù)兩者中最小值的整數(shù)倍。

有點繞,比如stu1 結(jié)構(gòu)體中,變量 c 類型為 int,也就是占用 4 字節(jié),編譯器采用 4 字節(jié)對齊,因此偏移量必須是 4 的整數(shù)倍。

typedef structstu2{    int a;    char b;    char c    int d;}stu2;

再比上面的 stu2中,如對于變量 c,其類型為 char ,占用 1 字節(jié),編譯器采用 4 字節(jié)對齊,因此 它被分配的偏移量需要是 1 的整數(shù)倍,在上面的結(jié)構(gòu)體 stu2 中,c 的偏移量為 5。

如圖:

圖片

現(xiàn)在來分析前面的代碼

假定a的起始地址為0,它占用了4字節(jié),接下來的空閑地址就是4,是1的倍數(shù),滿足要求,所以b存放的起始地址是4,占用一個字節(jié),接下來的空閑地址為5。c也是char變量,占用1字節(jié), 因此可以放在地址 5 上面。

接下來看地址 6,對于 d,它占用了 4 個字節(jié),同時需要注意的是,編譯器默認按照結(jié)構(gòu)體中占有內(nèi)存最大的類型所占用的字節(jié)數(shù)進行字節(jié)對齊。在此結(jié)構(gòu)體中占用內(nèi)存最大的為整型,占用4字節(jié),所以在此取兩者的最小值4,6 并不是4的整數(shù)倍,所以向后移動,找到離6最近的8作為存放d的起始地址,d也占用4字節(jié),最后結(jié)構(gòu)體的大小為12。

需要注意的就是,變量b和 c后面2字節(jié)的存儲空間是由編譯器自動填充的,其中沒有存儲任何有用的信息。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • ARM
    ARM
    +關(guān)注

    關(guān)注

    134

    文章

    9209

    瀏覽量

    371112
  • 字節(jié)
    +關(guān)注

    關(guān)注

    0

    文章

    41

    瀏覽量

    13888
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4857

    瀏覽量

    69540
  • 結(jié)構(gòu)體
    +關(guān)注

    關(guān)注

    1

    文章

    130

    瀏覽量

    10911
收藏 人收藏

    評論

    相關(guān)推薦

    C語言-結(jié)構(gòu)對齊詳解

    ", sizeof (struct mystruct1));// sizeof (struct mystruct1) = 8} 分析:整個結(jié)構(gòu)變量4字節(jié)對齊是由編譯器保證
    發(fā)表于 07-12 16:41

    RM48HDK平臺CCS結(jié)構(gòu)字節(jié)對齊總是咨詢

    硬件平臺:RM48HDK? 軟件平臺:5.4.0.00091? 編譯器:TI V5.0.6 我在程序中想對一個結(jié)構(gòu)增加單字節(jié)對齊屬性,增加方式如下: 1、參照《ARM Optimiz
    發(fā)表于 05-25 10:38

    CCS3.3 結(jié)構(gòu)成員對齊

    : expected a type specifier error: expected a ";" 這個錯誤。是CCS3.3編譯器本身不支持__packed這種結(jié)構(gòu)成員的對齊方式
    發(fā)表于 06-21 10:16

    請問在ccs4.2 中怎么設(shè)置結(jié)構(gòu)字節(jié)對齊

    請問在ccs4.2 中怎么設(shè)置結(jié)構(gòu)字節(jié)對齊,用于網(wǎng)絡(luò)數(shù)據(jù)發(fā)送的。是:]__attribute__((packed))還是:#pragma pack(1)編譯后,兩種都不行,是什么原
    發(fā)表于 08-02 07:47

    請問z-stack結(jié)構(gòu)默認對齊方式是一字節(jié)嗎?

    z-stack的結(jié)構(gòu)默認對齊方式是一字節(jié)嗎?在z-stack中可以將一般指針強制轉(zhuǎn)換為結(jié)構(gòu)
    發(fā)表于 08-18 07:38

    請問cc2640r2 ccs7.4結(jié)構(gòu)字節(jié)能實現(xiàn)對齊嗎?

    請問結(jié)構(gòu)字節(jié)下面能實現(xiàn)字節(jié)對齊么#pragma pack(1)typedef struct [ uint16_t nt ; uint8_t
    發(fā)表于 10-31 10:10

    STM32終極字節(jié)對齊的相關(guān)資料推薦

    字節(jié)對齊。uint64_t定義變量地址要8字節(jié)對齊。指針變量是4字節(jié)對齊。二、
    發(fā)表于 12-06 06:03

    解析C語言結(jié)構(gòu)字節(jié)如何對齊

    01 默認字節(jié)對齊 C語言結(jié)構(gòu)字節(jié)對齊是老生常談的問題了,也是高頻面試題,現(xiàn)在我們來深入研究這
    的頭像 發(fā)表于 06-12 17:42 ?3205次閱讀

    C語言中Linux字節(jié)對齊的問題

    ,于是經(jīng)過排查,是因為傳遞消息的結(jié)構(gòu)沒有考慮字節(jié)對齊的問題。 隨手整理一下C語言中字節(jié)對齊的問
    的頭像 發(fā)表于 08-16 11:25 ?2582次閱讀
    C語言中Linux<b class='flag-5'>字節(jié)</b><b class='flag-5'>對齊</b>的問題

    STM32 終極字節(jié)對齊解析

    字節(jié)對齊。uint64_t定義變量地址要8字節(jié)對齊。指針變量是4字節(jié)對齊。二、
    發(fā)表于 11-23 18:06 ?11次下載
    STM32 終極<b class='flag-5'>字節(jié)</b><b class='flag-5'>對齊</b>解析

    結(jié)構(gòu)對齊理解上有點偏差

    總結(jié)一下: 結(jié)構(gòu)對齊不再是簡單的字節(jié)個數(shù)的拼湊,而是要與內(nèi)存地址進行掛鉤~一般我們也可以理解為內(nèi)存地址分配是多少字節(jié)的倍數(shù),就是多少直接
    的頭像 發(fā)表于 08-10 18:08 ?1283次閱讀
    對<b class='flag-5'>結(jié)構(gòu)</b><b class='flag-5'>體</b>的<b class='flag-5'>對齊</b>理解上有點偏差

    結(jié)構(gòu)對齊為什么那么重要?

    以前我也沒覺得懂得這個結(jié)構(gòu)對齊或者內(nèi)存對齊有多重要,直到已經(jīng)從事了嵌入式開發(fā)經(jīng)驗不斷積累,才慢慢體會到,這是一種很基礎(chǔ)的知識,就因為這個東西不常用,而出現(xiàn)相關(guān)的問題是非常致命的,排查
    的頭像 發(fā)表于 04-03 10:13 ?1481次閱讀

    為什么要結(jié)構(gòu)對齊?為什么結(jié)構(gòu)對齊那么重要?

    C語言結(jié)構(gòu)對齊問題,是面試必備問題。我參與招聘技術(shù)面試的時候,也喜歡問這個技術(shù)點。
    的頭像 發(fā)表于 05-26 14:10 ?1477次閱讀
    為什么要<b class='flag-5'>結(jié)構(gòu)</b><b class='flag-5'>體</b><b class='flag-5'>對齊</b>?為什么<b class='flag-5'>結(jié)構(gòu)</b><b class='flag-5'>體</b><b class='flag-5'>對齊</b>那么重要?

    嵌套的結(jié)構(gòu) 字節(jié)是如何對齊

    嵌套的結(jié)構(gòu),字節(jié)又是如何對齊的呢 先來看下面的代碼 typedef struct stu1 { char ary [ 5 ] ; int a; }stu1; typedef stru
    的頭像 發(fā)表于 11-20 16:01 ?639次閱讀
    嵌套的<b class='flag-5'>結(jié)構(gòu)</b><b class='flag-5'>體</b> <b class='flag-5'>字節(jié)</b>是如何<b class='flag-5'>對齊</b>的

    keil arm工程中結(jié)構(gòu)1字節(jié)對齊如何實現(xiàn)

    的默認對齊方式和規(guī)則 結(jié)構(gòu)在Keil Arm工程中的默認對齊方式是根據(jù)編譯器或者編譯器選項來定的。通常情況下,編譯器會按照平臺的特定對齊
    的頭像 發(fā)表于 01-05 14:40 ?4275次閱讀