91欧美超碰AV自拍|国产成年人性爱视频免费看|亚洲 日韩 欧美一厂二区入|人人看人人爽人人操aV|丝袜美腿视频一区二区在线看|人人操人人爽人人爱|婷婷五月天超碰|97色色欧美亚州A√|另类A√无码精品一级av|欧美特级日韩特级

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

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

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

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

電子工程師 ? 來(lái)源:編程學(xué)習(xí)總站 ? 作者:寫代碼的牛頓 ? 2021-06-12 17:42 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

01

默認(rèn)字節(jié)對(duì)齊

C語(yǔ)言結(jié)構(gòu)體字節(jié)對(duì)齊是老生常談的問(wèn)題了,也是高頻面試題,現(xiàn)在我們來(lái)深入研究這個(gè)問(wèn)題,徹底弄懂到底是怎么回事,給你一個(gè)結(jié)構(gòu)體定義和平臺(tái)機(jī)器位數(shù)就能手動(dòng)計(jì)算出結(jié)構(gòu)體占用字節(jié)數(shù),現(xiàn)在我們不使用宏#pragma pack,采用默認(rèn)字節(jié)對(duì)齊方式。

先拋出結(jié)論:

在一個(gè)結(jié)構(gòu)體中第一個(gè)成員變量放在偏移為0的位置,以后的變量都存儲(chǔ)在該變量占用字節(jié)數(shù)整數(shù)倍的地址上。

結(jié)構(gòu)體總大小,必須是內(nèi)部最大成員變量的整數(shù)倍,不足的補(bǔ)齊。

好了,現(xiàn)在我們直接寫個(gè)小程序驗(yàn)證并分析是否真是這樣一回事。

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

這里我們定義了兩個(gè)很簡(jiǎn)單的結(jié)構(gòu)體,short占用2個(gè)字節(jié),struct st我們一眼就知道大小了6個(gè)字節(jié),但是struct st2呢?筆者電腦是64位,那么long占用8個(gè)字節(jié),short占用2個(gè)字節(jié)。我們先來(lái)按照結(jié)論進(jìn)行分析,在struct st2中成員變量a1在偏移0處存儲(chǔ)且占用8個(gè)字節(jié),成員變量a2占用2個(gè)字節(jié),由于8是2的倍數(shù),所以a2在偏移8的位置存儲(chǔ),又因?yàn)橛薪Y(jié)論2。

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

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; }

編譯運(yùn)行輸出:

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

現(xiàn)在我們看一下輸出結(jié)果,struct st如我們所愿占用6個(gè)字節(jié)大小,struct st2也按照我們分析的一樣占用16個(gè)字節(jié)。我們?cè)诔绦蛑卸x了一個(gè)struct st2類型變量st_val2,從輸出中可以看出變量st_val2的a1成員變量和st_val2變量地址一樣,成員變量a2在偏移8處存儲(chǔ)(0x c0 = 0xb8 + 8)。一切如我們所愿,看起來(lái)好像挺簡(jiǎn)單的,我們知道C語(yǔ)言有豐富的數(shù)據(jù)類型,下面我們?cè)俣x一個(gè)更復(fù)雜的結(jié)構(gòu)體。

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

這個(gè)結(jié)構(gòu)體包含了大量數(shù)據(jù)類型成員變量,再?gòu)?fù)雜的結(jié)構(gòu)體也能按照我們的結(jié)論分析到底占用了幾個(gè)字節(jié)。

在struct st3中int型成員變量a1占用4個(gè)字節(jié),在偏移0處存儲(chǔ),char型成員變量a2占用2個(gè)字節(jié)那么應(yīng)該放在2的倍數(shù)地址處存儲(chǔ),a1已經(jīng)占用了4個(gè)字節(jié),所以a2應(yīng)該在偏移4的地址存儲(chǔ)。

short型成員變量a3占用2個(gè)字節(jié),也應(yīng)該放在2的倍數(shù)地址處存儲(chǔ),所以a3在偏移6的地址處存儲(chǔ),a2后面填充1個(gè)字節(jié)。

long型成員變量a4占用8個(gè)字節(jié),應(yīng)該放在8的倍數(shù)地址上存儲(chǔ),前面我們已經(jīng)知道a3在偏移6的地址處存儲(chǔ),且占用2個(gè)字節(jié)8 = 6 + 2,所以a4應(yīng)該在偏移8的地址處存儲(chǔ)。

最后一個(gè)char型成員變量a5占用一個(gè)字節(jié),那么a5在偏移16地址處存儲(chǔ)。

現(xiàn)在我們計(jì)算一下struct st3結(jié)構(gòu)體占用空間大小,從a5偏移出計(jì)算16 + 1 = 17。在struct st3中最大成員變量占用8個(gè)字節(jié),所以結(jié)構(gòu)體總大小應(yīng)該是8的倍數(shù),最后結(jié)構(gòu)體總大小是17 + 7 = 24,這里的7個(gè)字節(jié)在最后補(bǔ)齊。

我們依舊寫一個(gè)小程序輸出struct st3類型變量各個(gè)成員變量地址和結(jié)構(gòu)體總大小。

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; }

編譯運(yùn)行輸出:

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

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

枚舉類型變量和聯(lián)合體類型變量都可以作為結(jié)構(gòu)體的成員變量,在分析這些結(jié)構(gòu)體占用大小時(shí),分析方法和我們上面的一模一樣,只需要把內(nèi)部任何一種數(shù)據(jù)類型變量當(dāng)做一個(gè)普通變量看待即可,但是結(jié)構(gòu)體類型成員變量有點(diǎn)不一樣,它不適用于結(jié)論2,我們舉個(gè)例子。

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

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

最后我們寫一個(gè)程序驗(yàn)證一下是否如此。

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; }

編譯運(yùn)行輸出:

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宏,表示結(jié)構(gòu)體按1字節(jié)對(duì)齊。也就是說(shuō)結(jié)構(gòu)體變量st_val3總大小是內(nèi)部成員變量占用字節(jié)數(shù)總和,沒有字節(jié)填充。

現(xiàn)在編譯運(yùn)行如下:

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字節(jié),char型變量a2占用1個(gè)字節(jié),short型變量a3占用2個(gè)字節(jié),long型變量a4占用8個(gè)字節(jié),char型變量a5占用1個(gè)字節(jié),所以總大小是:4 + 1 + 2 + 8 + 1 = 16。如果是#pagma pack(2)呢?相信你可以自己計(jì)算了。

編輯:jq

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

    關(guān)注

    183

    文章

    7644

    瀏覽量

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

    關(guān)注

    30

    文章

    4967

    瀏覽量

    73960

原文標(biāo)題:C語(yǔ)言結(jié)構(gòu)體字節(jié)對(duì)齊

文章出處:【微信號(hào):AndroidPush,微信公眾號(hào):Android編程精選】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    C語(yǔ)言主要特點(diǎn)

    。 3.數(shù)據(jù)類型豐富。C語(yǔ)言提供的數(shù)據(jù)類型包括整形、浮點(diǎn)型、字符型、數(shù)組類型、指針類型、結(jié)構(gòu)類型和共用類型等,
    發(fā)表于 01-05 07:41

    10個(gè)例子代碼,C語(yǔ)言結(jié)構(gòu)的高級(jí)

    當(dāng)涉及到C語(yǔ)言結(jié)構(gòu)的高級(jí)**時(shí),有很多有趣和強(qiáng)大的技巧可以應(yīng)用。下面是10個(gè)例子代碼,每個(gè)例子都使用了不同的高級(jí)結(jié)構(gòu)
    發(fā)表于 01-05 06:32

    單片機(jī)c語(yǔ)言中定義字節(jié)怎么定義?

    定義位可以用***it但是定義字節(jié)呢?還有就是比如匯編中的DPTR的DPH和DPL在c語(yǔ)言中怎樣定義的呢?
    發(fā)表于 12-29 06:32

    一個(gè)經(jīng)典的結(jié)構(gòu)和聯(lián)合體共用的實(shí)例

    結(jié)構(gòu) 結(jié)構(gòu)占用的內(nèi)存大小,首先和編譯器的系統(tǒng)位數(shù)有關(guān)系,類似于CPU是 64 bits 還是 32 bits 的情形;其次,結(jié)構(gòu)
    發(fā)表于 12-16 07:14

    結(jié)構(gòu)聲明與定義

    一樣,去聲明我們的結(jié)構(gòu)體操作,但是C語(yǔ)言中規(guī)定,聲明結(jié)構(gòu)變量的時(shí)候,struct關(guān)鍵字是不可少的。 struct
    發(fā)表于 12-11 07:52

    typedef結(jié)構(gòu)使用

    雖然結(jié)構(gòu)的出現(xiàn)能夠讓我們有一個(gè)更科學(xué)的數(shù)據(jù)結(jié)構(gòu)來(lái)管理數(shù)據(jù),但是每次使用結(jié)構(gòu)都需要struct...,未免顯得有些冗長(zhǎng)和麻煩。有了type
    發(fā)表于 12-08 07:04

    C語(yǔ)言的編程技巧

    一個(gè)成員是一個(gè)未知大小的數(shù)組,適用于動(dòng)態(tài)分配內(nèi)存并關(guān)聯(lián)一個(gè)可變長(zhǎng)度的數(shù)組。? ?3、匿名結(jié)構(gòu)和聯(lián)合體?:C語(yǔ)言允許在結(jié)構(gòu)
    發(fā)表于 11-27 06:46

    C語(yǔ)言程序的結(jié)構(gòu)

    ,87LPC764有4KB的Flash ROM,而筆者的程序量只有2KB多點(diǎn),因而第一個(gè)想法是改用C語(yǔ)言作為主要的開發(fā)語(yǔ)言,應(yīng)該不至于導(dǎo)致代碼空間不夠用。其次,考慮到需要定時(shí)功能的模塊(或稱任務(wù),以下統(tǒng)稱任務(wù)
    發(fā)表于 11-26 08:12

    C語(yǔ)言的分支結(jié)構(gòu)介紹

    1.簡(jiǎn)單if語(yǔ)句 C語(yǔ)言中的分支結(jié)構(gòu)語(yǔ)句中的if條件語(yǔ)句。 簡(jiǎn)單if語(yǔ)句的基本結(jié)構(gòu)如下: 代碼語(yǔ)言:javascript if(表達(dá)
    發(fā)表于 11-25 07:48

    C語(yǔ)言結(jié)構(gòu)使用

    型的數(shù)據(jù)組合。 結(jié)構(gòu)的聲明與定義 結(jié)構(gòu)的聲明一般形式為: c struct 結(jié)構(gòu)
    發(fā)表于 11-12 08:30

    MDK uVision V5.36.00使用rt_packed進(jìn)行字節(jié)對(duì)齊,但無(wú)效,為什么?

    各位RTT專家好: MDK uVision V5.36.00使用rt_packed進(jìn)行字節(jié)對(duì)齊,但無(wú)效。
    發(fā)表于 09-24 06:37

    《ESP32S3 Arduino開發(fā)指南》第三章 C/C++語(yǔ)言基礎(chǔ)

    ++基礎(chǔ),由于篇幅有限,在此僅對(duì)C/C++語(yǔ)言基礎(chǔ)進(jìn)行簡(jiǎn)單介紹。本章將分為如下9個(gè)小節(jié):3.1 數(shù)據(jù)類型3.2 運(yùn)算符3.3 表達(dá)式3.4 數(shù)組3.5 字符串3.6 注釋3.7 順序結(jié)構(gòu)
    發(fā)表于 06-10 09:20

    深入理解C語(yǔ)言C語(yǔ)言循環(huán)控制

    C語(yǔ)言編程中,循環(huán)結(jié)構(gòu)是至關(guān)重要的,它可以讓程序重復(fù)執(zhí)行特定的代碼塊,從而提高編程效率。然而,為了避免程序進(jìn)入無(wú)限循環(huán),C語(yǔ)言提供了多種循
    的頭像 發(fā)表于 04-29 18:49 ?2043次閱讀
    深入理解<b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>:<b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>循環(huán)控制

    C語(yǔ)言結(jié)構(gòu)與聯(lián)合體的深度解析:內(nèi)存布局與應(yīng)用場(chǎng)景

    一、基礎(chǔ)概念與核心差異 1.1 結(jié)構(gòu)(Struct)的本質(zhì) **結(jié)構(gòu)C語(yǔ)言中實(shí)現(xiàn)數(shù)據(jù)封裝的
    發(fā)表于 04-08 09:18

    全套C語(yǔ)言培訓(xùn)資料—PPT課件

    全套C語(yǔ)言培訓(xùn)資料,共427頁(yè),13個(gè)章節(jié):C語(yǔ)言概述、程序的靈魂—算法、數(shù)據(jù)類型 & 運(yùn)算符與表達(dá)式、順序程序設(shè)計(jì)、選擇結(jié)構(gòu)程序設(shè)
    發(fā)表于 03-12 14:50