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

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

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

關(guān)于C語言結(jié)構(gòu)體內(nèi)存對齊

Q4MP_gh_c472c21 ? 來源:嵌入式大雜燴 ? 作者:嵌入式大雜燴 ? 2022-04-14 12:51 ? 次閱讀

今天給大家?guī)硪坏澜?jīng)典、易錯的關(guān)于C語言結(jié)構(gòu)體內(nèi)存對齊的題目:

32bit環(huán)境下以下結(jié)構(gòu)體所占的字節(jié)數(shù):
typedefstructtest_struct
{
chara;
shortb;
charc;
intd;
chare;
}test_struct;

請說出你的答案:

下面看一下實際測試情況:

1、測試代碼:

/***********************************
*公眾號:嵌入式大雜燴
***********************************/
#include

typedefstructtest_struct
{
chara;
shortb;
charc;
intd;
chare;
}test_struct;

intmain(void)
{
test_structtest_s;

printf("
============================================
");
printf("test_saddr=%#.8x
",&test_s);
printf("test_s.aaddr=%#.8x
",&test_s.a);
printf("test_s.baddr=%#.8x
",&test_s.b);
printf("test_s.caddr=%#.8x
",&test_s.c);
printf("test_s.daddr=%#.8x
",&test_s.d);
printf("test_s.eaddr=%#.8x
",&test_s.e);
printf("sizeof(test_s)=%d
",sizeof(test_s));
printf("============================================
");

return0;
}

2、運行結(jié)果

關(guān)于C語言結(jié)構(gòu)體內(nèi)存對齊

在32bit環(huán)境中,該結(jié)構(gòu)體所占的字節(jié)數(shù)為16。答對了嗎?

運行結(jié)果打印輸出了很多重要的信息,從結(jié)果往前分析思路應該很清晰了吧?

下面,我們一起來分析分析。

3、分析

在分析這個問題之前,我們先記住關(guān)于結(jié)構(gòu)體內(nèi)存對齊的三條原則:

(1)結(jié)構(gòu)體變量的起始地址能夠被其最寬的成員大小整除。

(2)結(jié)構(gòu)體每個成員相對于起始地址的偏移能夠被其自身大小整除,如果不能則在前一個成員后面補充字節(jié)。

(3)結(jié)構(gòu)體總體大小能夠被最寬的成員的大小整除,如不能則在后面補充字節(jié)。

分析這個問題我們就不考慮編譯器可以指定對齊大小的情況了。在32bit環(huán)境中,一般默認的對齊大小是4。

下面,我們根據(jù)這三條原則來分析,并得出如下示意圖:

關(guān)于C語言結(jié)構(gòu)體內(nèi)存對齊

從這張圖中,我們應該可以很清晰地看出整個結(jié)構(gòu)體變量的內(nèi)存占用情況。

如果還看不明白的朋友,可以閱讀下面的解釋(有點啰嗦,已經(jīng)看明白的就不用看了~):

從上例的結(jié)果中,我們結(jié)構(gòu)體變量test_s的起始地址為0x0028ff30,能夠被其最寬的成員(int類型的d成員,占4個字節(jié))整除,符合第(1)條原則。

a成員的地址即為結(jié)構(gòu)體變量的起始地址0x0028ff30,排在a后面的是short類型(兩個字節(jié))的b成員。

根據(jù)第(2)條規(guī)則,顯然b的地址不能從0x0028ff31開始,則編譯器會在b成員的前一個成員(a成員)后邊補1個空白字節(jié),即b的的地址為從0x0028ff32,符合規(guī)則(2)。

b成員占兩個字節(jié),兩個字節(jié)之后的地址為0x0028ff34,而c成員為char類型(1字節(jié)),則根據(jù)規(guī)則(2),c成員會存放至地址0x0028ff34處。

c成員占1個字節(jié),1個字節(jié)之后的地址為0x0028ff35,排在c后面的是int類型(4個字節(jié))的d成員,顯然不能滿足規(guī)則(2)。

編譯器會在d成員的前一個成員(c成員)后面進行字節(jié)填充,這里必須填充3個字節(jié)才能符合規(guī)則(2),此時d會存放至地址0x0028ff38處。

d成員占4個字節(jié),4個字節(jié)之后的地址為0x0028ff3c。根據(jù)規(guī)則(2),e成員可從該地址開始存放。

此時a+空白字節(jié)+b+c+空白字節(jié)+d+e所占的字節(jié)總數(shù)為13個字節(jié),而結(jié)構(gòu)體最寬的成員(int類型的d成員)所占字節(jié)數(shù)為4字節(jié)。

顯然不能滿足規(guī)則(3),編譯器會在e成員后面填充3個字節(jié)。即整個結(jié)構(gòu)體變量test_s所占的總字節(jié)數(shù)為16字節(jié)。

4、實際應用

(1)用保留變量替代填充字節(jié)

實際應用中,我們可以上面的結(jié)構(gòu)體變量改為:

typedefstructtest_struct
{
chara;
charreserve0;/*保留成員*/
shortb;
charc;
intd;
chare;
charreserve1[3];/*保留成員*/
}test_struct;

我們已經(jīng)知道了編譯器會自動給我們的結(jié)構(gòu)體變量填充一些空白字節(jié),這些填充字節(jié)我們是看不到的,是隱性的。

在結(jié)構(gòu)體變量占用相同內(nèi)存的情況下,我們可以顯性的表示出這些填充字節(jié),即創(chuàng)建一些保留成員。

這樣,當我們需要給這個結(jié)構(gòu)體添加一些成員時,我們可以把保留的成員替換為實際的成員。這樣在一定程度下有利于我們節(jié)省內(nèi)存空間。

(2)調(diào)整結(jié)構(gòu)體成員的位置

從上面的分析中,我們知道編譯器會根據(jù)我們結(jié)構(gòu)體成員的排列來進行空白字節(jié)填充以達到對齊的效果。

那么,我們自己進行手動對齊一些成員,那就可以節(jié)省一些空間了。比如把上面的我們的test_struct結(jié)構(gòu)體成員的順序改為:

typedefstructtest_struct
{
chara;
charc;
shortb;
intd;
chare;
}test_struct;

則結(jié)構(gòu)體變量test_s所占的字節(jié)數(shù)變?yōu)?2字節(jié),即:

關(guān)于C語言結(jié)構(gòu)體內(nèi)存對齊

即比原來的16字節(jié)省下了4個字節(jié)。

雖然這點優(yōu)化對于一般的嵌入式應用來說可能沒什么必要,但是萬一某一天真的需要在某些資源極其受限的嵌入式設(shè)備中開發(fā)應用,這就是可以優(yōu)化的一點。

最后

以上就是本次的實驗分享。如有錯誤,歡迎指出!謝謝

這道結(jié)構(gòu)體內(nèi)存對齊的題目很經(jīng)典、也很容易出錯,是嵌入式C語言筆試、面試題中的高頻題目,很有必要弄清楚。

原文標題:嵌入式C結(jié)構(gòu)體內(nèi)存對齊

文章出處:【微信公眾號:嵌入式ARM】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

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

    關(guān)注

    8

    文章

    2902

    瀏覽量

    73536
  • C語言
    +關(guān)注

    關(guān)注

    180

    文章

    7575

    瀏覽量

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

    關(guān)注

    1

    文章

    127

    瀏覽量

    10800

原文標題:嵌入式C結(jié)構(gòu)體內(nèi)存對齊

文章出處:【微信號:gh_c472c2199c88,微信公眾號:嵌入式微處理器】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    關(guān)于C語言結(jié)構(gòu)體內(nèi)存對齊

    今天給大家?guī)硪坏澜?jīng)典、易錯的關(guān)于C語言結(jié)構(gòu)體內(nèi)存對齊的題目。
    發(fā)表于 09-08 11:54 ?431次閱讀

    C語言結(jié)構(gòu)對齊介紹

    大家好,我是嵌入式老林,從事嵌入式軟件開發(fā)多年,今天分享的內(nèi)容是C語言結(jié)構(gòu)對齊介紹,希望能對你有所幫助
    發(fā)表于 07-11 11:50 ?2170次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b><b class='flag-5'>結(jié)構(gòu)</b>體<b class='flag-5'>對齊</b>介紹

    C語言中數(shù)組和結(jié)構(gòu)體的內(nèi)存表示和布局

    C語言中,數(shù)組和結(jié)構(gòu)體都可以代表一塊內(nèi)存,但為什么結(jié)構(gòu)體可以直接賦值,而數(shù)組不可以?這個問題涉及到C
    發(fā)表于 08-28 10:54 ?1161次閱讀

    C語言零基礎(chǔ)教程之結(jié)構(gòu)體內(nèi)存對齊,手把手帶你輕松上手C語言

    C語言編程語言
    電子學習
    發(fā)布于 :2023年01月14日 13:22:17

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

    `C語言-結(jié)構(gòu)對齊詳解朱有鵬1、結(jié)構(gòu)體為何要對齊訪問訪問結(jié)
    發(fā)表于 07-12 16:41

    漫談C語言結(jié)構(gòu)

    我放在下面。  在此,我會圍繞以下2個問題來分析和應用C語言結(jié)構(gòu)體:  1. C語言中的結(jié)構(gòu)體有
    發(fā)表于 11-15 15:59

    掌握在單片機下使用C語言編程內(nèi)存對齊的知識點

    一、前言在單片機下使用C語言編程時,內(nèi)存對齊的知識點必須掌握。掌握內(nèi)存對齊后,可以防止
    發(fā)表于 02-28 07:47

    詳解C語言字節(jié)對齊

      一、什么是對齊,以及為什么要對齊:   1. 現(xiàn)代計算機中內(nèi)存空間都是按照byte劃分的,從理論上講似乎對任何類型的變量的訪問可以從任何地址開始,但實際情況是
    發(fā)表于 08-29 10:11 ?1208次閱讀

    了解內(nèi)存:如何在嵌入式C語言中使用結(jié)構(gòu)

    處理器如何訪問內(nèi)存?了解有關(guān)C語言結(jié)構(gòu)以及如何使用它們的更多信息。 本文將首先解釋內(nèi)存訪問粒度的概念,以便我們可以對處理器如何訪問
    的頭像 發(fā)表于 09-28 09:46 ?1510次閱讀

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

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

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

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

    嵌入式系統(tǒng)編程的大小端和內(nèi)存對齊問題

    C語言是一種高級語言,在大多數(shù)情況下C語言的代碼是和具體的處理器體系結(jié)構(gòu)無關(guān)的。然而,在嵌入式系
    的頭像 發(fā)表于 08-27 10:54 ?3165次閱讀

    C語言 | 內(nèi)存對齊01 - 什么是內(nèi)存對齊

    一、前言在單片機下使用C語言編程時,內(nèi)存對齊的知識點必須掌握。掌握內(nèi)存對齊后,可以防止
    發(fā)表于 01-13 15:18 ?0次下載
    <b class='flag-5'>C</b><b class='flag-5'>語言</b> | <b class='flag-5'>內(nèi)存</b><b class='flag-5'>對齊</b>01 - 什么是<b class='flag-5'>內(nèi)存</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 ?1124次閱讀
    對<b class='flag-5'>結(jié)構(gòu)</b>體的<b class='flag-5'>對齊</b>理解上有點偏差

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

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