在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

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

3天內不再提示

解析C語言結構體字節如何對齊

電子工程師 ? 來源:編程學習總站 ? 作者:寫代碼的牛頓 ? 2021-06-12 17:42 ? 次閱讀

01

默認字節對齊

C語言結構體字節對齊是老生常談的問題了,也是高頻面試題,現在我們來深入研究這個問題,徹底弄懂到底是怎么回事,給你一個結構體定義和平臺機器位數就能手動計算出結構體占用字節數,現在我們不使用宏#pragma pack,采用默認字節對齊方式。

先拋出結論:

在一個結構體中第一個成員變量放在偏移為0的位置,以后的變量都存儲在該變量占用字節數整數倍的地址上。

結構體總大小,必須是內部最大成員變量的整數倍,不足的補齊。

好了,現在我們直接寫個小程序驗證并分析是否真是這樣一回事。

struct st{ short a1; short a2; short a3; }; struct st2{ long a1; short a2; };

這里我們定義了兩個很簡單的結構體,short占用2個字節,struct st我們一眼就知道大小了6個字節,但是struct st2呢?筆者電腦是64位,那么long占用8個字節,short占用2個字節。我們先來按照結論進行分析,在struct st2中成員變量a1在偏移0處存儲且占用8個字節,成員變量a2占用2個字節,由于8是2的倍數,所以a2在偏移8的位置存儲,又因為有結論2。

我們根據結論2可以得出,struct st2必須占用8的倍數大小,所以struct st2總大小是16個字節,不足的后面補齊。現在我分別打印出struct st1和struct st2占用字節數大小和struct st2各個成員變量地址,觀察是否和分析的一樣。

int main() { struct st2 st_val2; printf(“sizeof(long) = %d ”, sizeof(long)); printf(“sizeof(struct st) = %d ”, sizeof(struct st)); printf(“sizeof(struct st2) = %d ”, sizeof(struct st2)); printf(“st_val2 addr = %p ”, &st_val2); printf(“st_val2 a1 addr = %p ”, &st_val2.a1); printf(“st_val2 a2 addr = %p ”, &st_val2.a2); return 0; }

編譯運行輸出:

sizeof(long) = 8 sizeof(struct st) = 6 sizeof(struct st2) = 16 st_val2 addr = 0x7ffee107f3b8 st_val2 a1 addr = 0x7ffee107f3b8 st_val2 a2 addr = 0x7ffee107f3c0

現在我們看一下輸出結果,struct st如我們所愿占用6個字節大小,struct st2也按照我們分析的一樣占用16個字節。我們在程序中定義了一個struct st2類型變量st_val2,從輸出中可以看出變量st_val2的a1成員變量和st_val2變量地址一樣,成員變量a2在偏移8處存儲(0x c0 = 0xb8 + 8)。一切如我們所愿,看起來好像挺簡單的,我們知道C語言有豐富的數據類型,下面我們再定義一個更復雜的結構體。

struct st3{ int a1; char a2; short a3; long a4; char a5; };

這個結構體包含了大量數據類型成員變量,再復雜的結構體也能按照我們的結論分析到底占用了幾個字節。

在struct st3中int型成員變量a1占用4個字節,在偏移0處存儲,char型成員變量a2占用2個字節那么應該放在2的倍數地址處存儲,a1已經占用了4個字節,所以a2應該在偏移4的地址存儲。

short型成員變量a3占用2個字節,也應該放在2的倍數地址處存儲,所以a3在偏移6的地址處存儲,a2后面填充1個字節。

long型成員變量a4占用8個字節,應該放在8的倍數地址上存儲,前面我們已經知道a3在偏移6的地址處存儲,且占用2個字節8 = 6 + 2,所以a4應該在偏移8的地址處存儲。

最后一個char型成員變量a5占用一個字節,那么a5在偏移16地址處存儲。

現在我們計算一下struct st3結構體占用空間大小,從a5偏移出計算16 + 1 = 17。在struct st3中最大成員變量占用8個字節,所以結構體總大小應該是8的倍數,最后結構體總大小是17 + 7 = 24,這里的7個字節在最后補齊。

我們依舊寫一個小程序輸出struct st3類型變量各個成員變量地址和結構體總大小。

int main() { struct st3 st_val3; printf(“sizeof(struct st3) = %d ”, sizeof(struct st3)); printf(“st_val3 addr = %p ”, &st_val3); printf(“st_val3.a1 addr = %p ”, &st_val3.a1); printf(“st_val3.a2 addr = %p ”, &st_val3.a2); printf(“st_val3.a3 addr = %p ”, &st_val3.a3); printf(“st_val3.a4 addr = %p ”, &st_val3.a4); printf(“st_val3.a5 addr = %p ”, &st_val3.a5); return 0; }

編譯運行輸出:

sizeof(struct st3) = 24 st_val3 addr = 0x7ffeed0c33b0 st_val3.a1 addr = 0x7ffeed0c33b0 st_val3.a2 addr = 0x7ffeed0c33b4 st_val3.a3 addr = 0x7ffeed0c33b6 st_val3.a4 addr = 0x7ffeed0c33b8 st_val3.a5 addr = 0x7ffeed0c33c0

從輸出我們可以看出,和我們分析的完全一樣。

枚舉類型變量和聯合體類型變量都可以作為結構體的成員變量,在分析這些結構體占用大小時,分析方法和我們上面的一模一樣,只需要把內部任何一種數據類型變量當做一個普通變量看待即可,但是結構體類型成員變量有點不一樣,它不適用于結論2,我們舉個例子。

struct st4{ char a1[3]; int a2; long a3; struct st3 a4; };

在struct st4中我們定義了一個struct st3類型成員變量,前面我們已經分析過了struct st3占用24個字節。成員變量a1占用3個字節,成員變量a2占用4個字節,所以a2存儲在偏移4的地址上,在a1后面填充一個字節。成員變量a3占用8個字節,則a3存儲在偏移8的地址上。那么結構體總共占用字節數大小是:8 + 8 + 24 = 40。

最后我們寫一個程序驗證一下是否如此。

int main() { struct st4 st_val4; printf(“sizeof(struct st4) = %d ”, sizeof(struct st4)); printf(“st4 addr = %p ”, &st_val4); printf(“st_val4.a1 addr = %p ”, &st_val4.a1); printf(“st_val4.a2 addr = %p ”, &st_val4.a2); printf(“st_val4.a3 addr = %p ”, &st_val4.a3); printf(“st_val4.a4 addr = %p ”, &st_val4.a4); return 0; }

編譯運行輸出:

sizeof(struct st4) = 40 st4 addr = 0x7ffeec1263a0 st_val4.a1 addr = 0x7ffeec1263a0 st_val4.a2 addr = 0x7ffeec1263a4 st_val4.a3 addr = 0x7ffeec1263a8 st_val4.a4 addr = 0x7ffeec1263b0

和我們分析的一模一樣。

02

#pragma pack宏的作用

我們看一下下面這段代碼。

#pagma pack(1)int main() { struct st3 st_val3; printf(“sizeof(struct st3) = %d ”, sizeof(struct st3)); printf(“st_val3 addr = %p ”, &st_val3); printf(“st_val3.a1 addr = %p ”, &st_val3.a1); printf(“st_val3.a2 addr = %p ”, &st_val3.a2); printf(“st_val3.a3 addr = %p ”, &st_val3.a3); printf(“st_val3.a4 addr = %p ”, &st_val3.a4); printf(“st_val3.a5 addr = %p ”, &st_val3.a5); return 0; }

這段代碼里我們使用了#pagma pack宏,表示結構體按1字節對齊。也就是說結構體變量st_val3總大小是內部成員變量占用字節數總和,沒有字節填充。

現在編譯運行如下:

sizeof(struct st3) = 16 st_val3 addr = 0x7ffee13a93b8 st_val3.a1 addr = 0x7ffee13a93b8 st_val3.a2 addr = 0x7ffee13a93bc st_val3.a3 addr = 0x7ffee13a93bd st_val3.a4 addr = 0x7ffee13a93bf st_val3.a5 addr = 0x7ffee13a93c7

在struct st3中int型a1占用4字節,char型變量a2占用1個字節,short型變量a3占用2個字節,long型變量a4占用8個字節,char型變量a5占用1個字節,所以總大小是:4 + 1 + 2 + 8 + 1 = 16。如果是#pagma pack(2)呢?相信你可以自己計算了。

編輯:jq

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • C語言
    +關注

    關注

    180

    文章

    7604

    瀏覽量

    136841
  • 代碼
    +關注

    關注

    30

    文章

    4788

    瀏覽量

    68616

原文標題:C語言結構體字節對齊

文章出處:【微信號:AndroidPush,微信公眾號:Android編程精選】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    結構成員的順序會影響結構的大小嗎

    相同的結構成員,如果把順序調整一下,會不會影響結構的大小? 答案是會的,這主要跟字節對齊有關
    的頭像 發表于 11-25 16:24 ?175次閱讀

    C語言C++中結構的區別

    同樣是結構,看看在C語言C++中有什么區別?
    的頭像 發表于 10-30 15:11 ?228次閱讀

    字節豆包推出AI智能耳機

    字節跳動旗下豆包品牌近日推出了其首款AI智能耳機——Ola Friend。這款耳機采用了開放式設計,單耳重量僅為6.6克,佩戴舒適輕盈。
    的頭像 發表于 10-10 16:55 ?528次閱讀

    技術干貨驛站 ▏深入理解C語言:編程高手必備,全方位解析運算符的核心技能!

    C語言的編程領域中,運算符是實現數據處理與邏輯操作的關鍵工具。無論是在處理簡單的數值計算,還是在構建復雜的邏輯結構時,運算符的使用貫穿始終。作為編程語言的基礎組成部分之一,運算符不僅
    的頭像 發表于 09-18 15:56 ?355次閱讀
    技術干貨驛站 ▏深入理解<b class='flag-5'>C</b><b class='flag-5'>語言</b>:編程高手必備,全方位<b class='flag-5'>解析</b>運算符的核心技能!

    技術干貨驛站 ▏深入理解C語言:掌握程序結構知識

    在計算機編程的世界中,C語言被廣泛認可為一門強大而高效的編程語言,其簡潔的語法和直接的指令使得它成為了許多程序員的首選。了解C語言的程序
    的頭像 發表于 07-27 08:45 ?1385次閱讀
    技術干貨驛站 ▏深入理解<b class='flag-5'>C</b><b class='flag-5'>語言</b>:掌握程序<b class='flag-5'>結構</b>知識

    你是否真的了解結構占用了多少字節

    結構成員所占內存空間大小一般情況下,如果想知道結構成員的內存占用情況需要:1、先用結構在內
    的頭像 發表于 06-04 08:04 ?464次閱讀
    你是否真的了解<b class='flag-5'>結構</b><b class='flag-5'>體</b>占用了多少<b class='flag-5'>字節</b>?

    嵌入式中C語言結構基本實現

    C語言中的數組只能允許程序員定義存儲相同類型數據。但是結構C語言編程中允許您存儲不同數據類型的數據。
    的頭像 發表于 05-11 08:49 ?1040次閱讀
    嵌入式中<b class='flag-5'>C</b><b class='flag-5'>語言</b><b class='flag-5'>結構</b><b class='flag-5'>體</b>基本實現

    求助,關于STM32H7的Cache無效化操作32字節對齊問題求解

    core_cm7.h更新到5.1.1版本后,發現該函數對dsize做了32字節對齊,但是op_addr地址32字節對齊卻注釋掉了?,圖片的這句話是不是說SCB-&gt
    發表于 03-29 06:51

    C語言結構史上最詳細的講解【軟件干貨】

    struct結構數據類型 前言 我們知道,在C語言中有一些基本的數據類型,如?char?int?float?long?double?string(
    的頭像 發表于 03-28 17:52 ?763次閱讀

    嵌入式系統中C語言結構的基礎實現與應用

    C語言中的數組只能允許程序員定義存儲相同類型數據。但是結構C語言編程中允許您存儲不同數據類型的數據。
    發表于 03-12 14:29 ?502次閱讀
    嵌入式系統中<b class='flag-5'>C</b><b class='flag-5'>語言</b><b class='flag-5'>結構</b><b class='flag-5'>體</b>的基礎實現與應用

    C語言中的typedef的應用

    C 語言提供了 typedef 關鍵字,您可以使用它來為類型取一個新的名字。下面的實例為單字節數字定義了一個術語 BYTE。
    發表于 03-06 11:34 ?387次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>中的typedef的應用

    ArkTS語言基礎類庫-解析

    被設計用來傳輸和存儲數據,是一種可擴展標記語言語言基礎類庫提供了[XML生成、解析與轉換]的能力。 URL、URI構造和解析能力:其中[URI]是統一資源標識符,可以唯一標識一個資源
    發表于 02-20 16:44

    經典 C 語言編程,結構和聯合體如何共用?

    結構 結構占用的內存大小,首先和編譯器的系統位數有關系,類似于CPU是 64 bits 還是 32 bits 的情形;其次,結構
    的頭像 發表于 01-11 18:24 ?1377次閱讀
    經典 <b class='flag-5'>C</b> <b class='flag-5'>語言</b>編程,<b class='flag-5'>結構</b><b class='flag-5'>體</b>和聯合體如何共用?

    結構與指針的關系

    C語言中,結構(Struct)是一種用戶自定義的數據類型,它允許您將不同類型的數據項組合在一起,以便形成一個更復雜的數據結構
    的頭像 發表于 01-11 08:00 ?1000次閱讀
    <b class='flag-5'>結構</b><b class='flag-5'>體</b>與指針的關系

    keil arm工程中結構1字節對齊如何實現

    的默認對齊方式和規則 結構在Keil Arm工程中的默認對齊方式是根據編譯器或者編譯器選項來定的。通常情況下,編譯器會按照平臺的特定對齊
    的頭像 發表于 01-05 14:40 ?3812次閱讀
    主站蜘蛛池模板: 韩国一级网站| 国产98色在线| 久久综合九色欧美综合狠狠| 美女毛片视频| 国产aaaaaa| 波多野结衣久久国产精品| 国产黄色小视频网站| 日本欧美一区二区三区视频| 伊人久久大杳蕉综合大象| 亚洲色网址| 视频免费1区二区三区| 色国产精品| 四虎最新永久免费网址| 欧美片欧美日韩国产综合片| 国内黄色一级片| 欧美成人免费观看bbb| 男人cao女人视频在线观看| 国产精品第九页| 色老头综合免费视频| 在线亚洲小视频| 国模吧双双大尺度炮交gogo| 中文字幕一区二区三区5566| 亚洲色图综合网站| 欧美一区二区三区免费| 丁香六月婷婷在线| 免费高清视频在线观看| 一区二区三区亚洲视频| 特级毛片aaa免费版| 夜夜爱视频| 色爱综合区五月小说| 亚洲综合激情六月婷婷在线观看| 女人张开腿给人桶免费视频| 亚洲国产成人久久午夜| 2019天天干| 亚洲图片综合区另类图片| www.亚洲视频.com| 黑粗硬大欧美视频| 在线看你懂| 激情五月亚洲色图| 天堂网中文| 天天综合网网欲色|