亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

詳解C語言如何計算結(jié)構(gòu)體大小(結(jié)構(gòu)體的內(nèi)存對齊)

 更新時間:2023年07月21日 10:10:18   作者:可涵不會debug  
結(jié)構(gòu)體的內(nèi)存對齊是有關(guān)結(jié)構(gòu)體內(nèi)容的很重要一個知識點(diǎn),主要考察方式是計算結(jié)構(gòu)體的字節(jié)大小,所以本文就給大家詳細(xì)介紹一下C語言如何計算結(jié)構(gòu)體大小,文中的代碼示例介紹的非常詳細(xì),需要的朋友可以參考下

引言:

當(dāng)我們對計算結(jié)構(gòu)體一無所知,我們不妨自己思索如何計算,是不是直接計算結(jié)構(gòu)體成員變量占用內(nèi)存的大小呢?

那我們先舉個例子

struct s1
{
	int i;
	char a;
	char b;
};
struct s2
{
	char a;
	int i;
	char b;
};
int main()
{
	printf("%d\n", sizeof(struct s1));
	printf("%d\n", sizeof(struct s2));
	return 0;
}

觀察發(fā)現(xiàn)結(jié)構(gòu)體的大小計算跟我們想的很不一樣。

不應(yīng)該是兩個char類型,一個int類型,2*1+4答案不應(yīng)該是6嗎?

上面兩個結(jié)構(gòu)體內(nèi)容是一樣的,只有順序不一樣,為何計算結(jié)果不一樣呢?

我們就帶著以上的疑問去探索!

一、計算偏移量

我們要研究明白結(jié)構(gòu)體的成員列表在內(nèi)存中到底是如何存儲的,首先要知道結(jié)構(gòu)體的各個成員變量在內(nèi)存中相較于起始位置的偏移量。這時候要引用到offsetof,這個宏可以計算結(jié)構(gòu)體成員相較于結(jié)構(gòu)體起始位置的偏移量。

使用宏offsetof

 如何使用宏offsetof?

首先有頭文件:#include<stddef.h>

參數(shù)是類型,和成員名,返回值就是結(jié)構(gòu)體成員相較于結(jié)構(gòu)體起始位置的偏移量。

我們先試著打印下s2各個成員關(guān)于結(jié)構(gòu)體起始位置的偏移量。

發(fā)現(xiàn)結(jié)果是0、4、8,我們可以畫一張內(nèi)存圖進(jìn)行理解。

如圖所示,根據(jù)offsetof我們可以得到這樣的內(nèi)存存儲模式,但是這樣一共也就9個字節(jié),后面的3個字節(jié)從何而來?中間多出來的3個字節(jié)又從何而來?

我們繼續(xù)探索。

結(jié)構(gòu)體到底如何計算?

二、結(jié)構(gòu)體的對齊規(guī)則

我們經(jīng)過上面的分析,發(fā)現(xiàn)結(jié)構(gòu)體成員不是按照順序在內(nèi)存中連續(xù)存放的,而是有一定的對齊規(guī)則,接下來我們就研究結(jié)構(gòu)體的內(nèi)存規(guī)則。

  • 結(jié)構(gòu)體的第一個成員永遠(yuǎn)放在相較于結(jié)構(gòu)體變量的起始未知的偏移量為0的位置
  • 從第二個成員開始,往后的每個成員都要對齊到某個對齊數(shù)的整數(shù)倍處。(對齊數(shù):結(jié)構(gòu)體成員自身大小和默認(rèn)對齊數(shù)的較小值)VS上默認(rèn)對齊數(shù)是8,gcc沒有默認(rèn)對齊數(shù),對齊數(shù)就是變量本身的大小。
  • 結(jié)構(gòu)體的總大小,必須是最大對齊數(shù)的整數(shù)倍,最大對齊數(shù)是:所有成員的對齊數(shù)中最大的值
  • 如果嵌套了結(jié)構(gòu)體的情況,嵌套的結(jié)構(gòu)體對齊到自己的最大對齊數(shù)的整數(shù)倍處,結(jié)構(gòu)體的整體大小就是所有最大對齊數(shù)(含嵌套結(jié)構(gòu)體的對齊數(shù))的整數(shù)倍。

三、總結(jié)計算方法

我們首先要知道結(jié)構(gòu)體變量成員的自身字節(jié)大小,然后去尋找對齊數(shù),對齊數(shù)的尋找方法就是將自身字節(jié)大小和默認(rèn)對齊數(shù)比較,取較小值,這樣先找到對齊數(shù),然后根據(jù)自身的字節(jié)大小去填充,就完成了成員在內(nèi)存中的存儲,最后在所有的成員已經(jīng)結(jié)束存儲,再計算最大對齊數(shù)(所有成員的對齊數(shù)中最大值),這樣就完成了計算!

我們既然已經(jīng)知道規(guī)則和計算方法,就讓我們小試牛刀一下~

四、練習(xí)

練習(xí)一:

struct s3
{
	double d;
	char c;
	int i;
};
int main()
{
	printf("%d\n", sizeof(struct s3));
	return 0;
}

上面圖片的寫法就是左邊是本身成員變量的字節(jié)大小,右邊是默認(rèn)對齊數(shù)進(jìn)行比較,最后再從對齊數(shù)中找出最大值,就是最大對齊數(shù),所以最后0~15就是存儲結(jié)構(gòu)體的大小,也就是一共16個字節(jié)

 練習(xí)二:

struct S3
{
	double d;
	char c;
	int i;
};
struct S4
{
	char c1;
	struct S3 s3;
	double d;
};
int main()
{
	printf("%d\n", sizeof(struct S4));
	return 0;
}

上面是嵌套結(jié)構(gòu)體場景,結(jié)構(gòu)體S3本身大小是16,需要對齊到自身最大對齊數(shù)的位置,也就是8,然后double類型的對齊數(shù)是8,最后總字節(jié)大小也滿足最大對齊數(shù),所以一共32個字節(jié)。

五、為什么存在內(nèi)存對齊?

1、平臺原因

不是所有的硬件平臺都能訪問任意地址上的任意數(shù)據(jù);某些平臺只能在某些地址處取某些地址處取特定類型的數(shù)據(jù),否則拋出硬件異常

2、性能原因

數(shù)據(jù)結(jié)構(gòu)(尤其是棧)應(yīng)該盡可能在自然邊界上對齊。原因在于,為了訪問未對齊的內(nèi)存,處理器需要作兩次內(nèi)存訪問;而對齊的內(nèi)存訪問僅需要一次訪問。

總體來說

結(jié)構(gòu)體的內(nèi)存對齊,就是讓空間換時間。 

TIP:

我們在設(shè)計結(jié)構(gòu)體時,可以人為的節(jié)省空間——讓占用空間小的成員盡量集中在一起。

例如我們之前舉的例子,盡管兩個結(jié)構(gòu)體存的成員變量一樣,但是順序不一樣,結(jié)構(gòu)體內(nèi)存大小也是不同。

六、修改默認(rèn)對齊數(shù)

對,你沒有聽錯,默認(rèn)對齊數(shù)是可以修改滴,當(dāng)我們把默認(rèn)對齊數(shù)修改為1時,結(jié)構(gòu)體的成員變量就是連續(xù)存儲的。代碼如下,計算出來的大小就是4+1+8=13

#pragma pack(1)//修改默認(rèn)對齊數(shù)為1
struct s
{
	int a;
	char b;
	double c;
};
#pragma pack()//修改默認(rèn)對齊數(shù)為默認(rèn)
int main()
{
	printf("%d\n", sizeof(struct s));
	return 0;
}

到此這篇關(guān)于詳解C語言如何計算結(jié)構(gòu)體大小(結(jié)構(gòu)體的內(nèi)存對齊)的文章就介紹到這了,更多相關(guān)C語言計算結(jié)構(gòu)體大小內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C語言中改變目錄的相關(guān)操作函數(shù)詳解

    C語言中改變目錄的相關(guān)操作函數(shù)詳解

    這篇文章主要介紹了C語言中改變目錄的相關(guān)操作函數(shù)詳解,分別是fchdir()函數(shù)和rewinddir()函數(shù)的使用方法,需要的朋友可以參考下
    2015-09-09
  • C語言數(shù)據(jù)結(jié)構(gòu)之單鏈表存儲詳解

    C語言數(shù)據(jù)結(jié)構(gòu)之單鏈表存儲詳解

    鏈表是一種物理存儲結(jié)構(gòu)上非連續(xù)、非順序的存儲結(jié)構(gòu),數(shù)據(jù)元素的邏輯順序是通過鏈表中的指針鏈接次序?qū)崿F(xiàn)的。本文將和大家一起聊聊C語言中單鏈表的存儲,感興趣的可以學(xué)習(xí)一下
    2022-07-07
  • 詳解C語言中return與exit的區(qū)別

    詳解C語言中return與exit的區(qū)別

    這篇文章主要介紹了詳解C語言中return與exit的區(qū)別的相關(guān)資料,希望通過本文能幫助到大家,讓大家理解這部分內(nèi)容,需要的朋友可以參考下
    2017-10-10
  • C/C++ 中g(shù)cc和g++的對比與區(qū)別

    C/C++ 中g(shù)cc和g++的對比與區(qū)別

    這篇文章主要介紹了C/C++ 中g(shù)cc和g++的對比與區(qū)別的相關(guān)資料,需要的朋友可以參考下
    2017-07-07
  • 解析bitmap處理海量數(shù)據(jù)及其實現(xiàn)方法分析

    解析bitmap處理海量數(shù)據(jù)及其實現(xiàn)方法分析

    本篇文章是對bitmap處理海量數(shù)據(jù)及其實現(xiàn)的方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-05-05
  • C++ Boost Chrono實現(xiàn)計時碼表流程詳解

    C++ Boost Chrono實現(xiàn)計時碼表流程詳解

    Boost是為C++語言標(biāo)準(zhǔn)庫提供擴(kuò)展的一些C++程序庫的總稱。Boost庫是一個可移植、提供源代碼的C++庫,作為標(biāo)準(zhǔn)庫的后備,是C++標(biāo)準(zhǔn)化進(jìn)程的開發(fā)引擎之一,是為C++語言標(biāo)準(zhǔn)庫提供擴(kuò)展的一些C++程序庫的總稱
    2022-11-11
  • C語言實現(xiàn)類似wget的進(jìn)度條效果

    C語言實現(xiàn)類似wget的進(jìn)度條效果

    這篇文章主要介紹了C語言實現(xiàn)類似wget的進(jìn)度條效果的方法,主要是讓大家可以熟練的使用轉(zhuǎn)移符\r,這里推薦給大家,需要的小伙伴參考下。
    2015-03-03
  • C語言實現(xiàn)隨機(jī)抽獎程序

    C語言實現(xiàn)隨機(jī)抽獎程序

    這篇文章主要為大家詳細(xì)介紹了C語言實現(xiàn)隨機(jī)抽獎程序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-09-09
  • C語言數(shù)據(jù)結(jié)構(gòu)實例講解單鏈表的實現(xiàn)

    C語言數(shù)據(jù)結(jié)構(gòu)實例講解單鏈表的實現(xiàn)

    單鏈表是后面要學(xué)的雙鏈表以及循環(huán)鏈表的基礎(chǔ),要想繼續(xù)深入了解數(shù)據(jù)結(jié)構(gòu)以及C++,我們就要奠定好這塊基石!接下來就和我一起學(xué)習(xí)吧
    2022-03-03
  • Qt實現(xiàn)拖動單個控件移動的示例代碼

    Qt實現(xiàn)拖動單個控件移動的示例代碼

    做慣了靜態(tài)圖,今天來搞一搞動態(tài)圖吧!本文將利用Qt實現(xiàn)拖動單個控件移動效果,文中的示例代碼講解詳細(xì),感興趣的可以動手嘗試一下
    2022-06-06

最新評論