溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

C++中sizeof的作用是什么

發(fā)布時(shí)間:2020-08-17 10:25:13 來源:億速云 閱讀:207 作者:小新 欄目:開發(fā)技術(shù)

這篇文章給大家分享的是有關(guān)C++中sizeof的作用是什么的內(nèi)容。小編覺得挺實(shí)用的,因此分享給大家做個(gè)參考。一起跟隨小編過來看看吧。

sizeof作用于基本數(shù)據(jù)類型,在特定的平臺(tái)和特定的編譯器中,結(jié)果是確定的,如果使用sizeof計(jì)算構(gòu)造類型:結(jié)構(gòu)體、聯(lián)合體和類的大小時(shí),情況稍微復(fù)雜一些。

1.sizeof計(jì)算結(jié)構(gòu)體

考察如下代碼:

struct S1
{
char c;
int i;
};
cout<<”sizeof(S1)=”<<sizeof(S1)<<endl;

sizeof(S1)結(jié)果是8,并不是想象中的sizeof(char)+sizeof(int)=5。這是因?yàn)榻Y(jié)構(gòu)體或類成員變量具有不同類型時(shí),需進(jìn)行成員變量的對(duì)齊?!队?jì)算機(jī)組成原理》一書中說明,對(duì)齊的目的是減少訪存指令周期,提高CPU存儲(chǔ)速度。

1.1內(nèi)存對(duì)齊原則

(1)結(jié)構(gòu)體變量的首地址能夠被其最寬基本成員類型大小所整除;

(2)結(jié)構(gòu)體每個(gè)成員相對(duì)于結(jié)構(gòu)體首地址的偏移量都是成員大小的整數(shù)倍,如有需要編譯器會(huì)在成員之間加上填充字節(jié);

(3)結(jié)構(gòu)體的總大小為結(jié)構(gòu)體最寬基本成員類型大小的整數(shù)倍,如有需要編譯器會(huì)在最末一個(gè)成員之后加上填充字節(jié)。

有了以上三個(gè)內(nèi)存對(duì)齊的原則,就可以輕松應(yīng)對(duì)嵌套結(jié)構(gòu)體類型的內(nèi)存對(duì)齊。如下:

struct S2
{
char c1;
S1 s;
char c2;
};

在尋找S2的最寬基本數(shù)據(jù)類型時(shí),包括其嵌套的結(jié)構(gòu)體中的成員,從S1中尋找出最寬結(jié)構(gòu)體數(shù)據(jù)類型是int,因此S2的最寬數(shù)據(jù)類型是int。S1 s在結(jié)構(gòu)體S2中的對(duì)齊也遵守前三個(gè)準(zhǔn)則,因此sizeof(S2)=sizeof(char)+pad(3)+sizeof(S1)+1+pad(3)=1+3+8+1+3=16字節(jié),其中pad(3)表示填充3個(gè)字節(jié)。

結(jié)構(gòu)體某個(gè)成員相對(duì)于結(jié)構(gòu)體首地址的偏移量可以通過宏offsetof()來獲得,這個(gè)宏也在stddef.h中定義,如下:

#define offsetof(s,m) (size_t)&(((s *)0)->m)

例如獲得S1中的偏移量,方法為

size_t pos = offsetof(S1, i); //pos等于4

1.2修改對(duì)齊方式

1.2.1#pragma pack

#pragma pack(n)中n為字節(jié)對(duì)齊數(shù),其取值為1、2、4、8、16,默認(rèn)是8。結(jié)構(gòu)體對(duì)齊時(shí),

(1)成員的偏移量為成員本身大小和n二者最小值的整數(shù)倍;
(2)結(jié)構(gòu)體最終大小是結(jié)構(gòu)體中最寬基本類型成員大小和n二者中的最小值的整數(shù)倍。

考察如下代碼:

#pragma pack(push) //將當(dāng)前pack設(shè)置壓棧保存
#pragma pack(2) //必須在結(jié)構(gòu)體定義之前使用
struct S1
{
char c;
int i;
};
struct S2
{
char c1;
S1 s;
char c2
};
#pragma pack(pop) // 恢復(fù)先前的pack設(shè)置

//或者
#pragma pack(2)
...
#pragma pack()

因此,sizeof(S2)=sizeof(char)+pad(1)+sizeof(S1)+1+pad(1)=1+1+6+1=10字節(jié)。

注意,#pragma pack不能指定變量的存儲(chǔ)地址,變量的首地址默認(rèn)為最大基本成員類型大小的整數(shù)倍。

1.2.2__declspec(align(#))

VC++支持__declspec(align(#)),在GNU C++并不支持。#的取值為1~8192,為2的冪。使用示例如下:

__declspec(align(256)) struct TestSize
{
char a;
int i;
};
cout<<sizeof(TestSize)<<endl; //輸出256

__declspec(align(#))要求#為2的整數(shù)次冪,作用主要有兩個(gè)方面:
(1)使結(jié)構(gòu)體或類成員按#pragma pack確定內(nèi)存布局之后,在末尾填充內(nèi)存使得整個(gè)對(duì)象的大小至少是#的整數(shù)倍。
(2)作用于變量時(shí),強(qiáng)制要求編譯器將變量放置在地址是#整數(shù)倍的內(nèi)存位置上。這點(diǎn)在調(diào)用原生API等要求嚴(yán)格對(duì)齊的方法時(shí)十分重要。

1.3空結(jié)構(gòu)體

C/C++中不允許長(zhǎng)度為0的數(shù)據(jù)類型存在。對(duì)于“空結(jié)構(gòu)體”(不含數(shù)據(jù)成員)的大小不為0,而是1。“空結(jié)構(gòu)體”變量也得被存儲(chǔ),這樣編譯器也就只能為其分配一個(gè)字節(jié)的空間用于占位了。如下:

struct S3 { };
sizeof(S3); // 結(jié)果為1

1.4位域結(jié)構(gòu)體

有些信息在存儲(chǔ)時(shí),并不需要占用一個(gè)完整的字節(jié), 而只需占一個(gè)或多個(gè)二進(jìn)制位。例如在存放一個(gè)開關(guān)量時(shí),只有0和1 兩種狀態(tài), 用一位即可表示。為了節(jié)省存儲(chǔ)空間,并使處理簡(jiǎn)便,C語言又提供了一種數(shù)據(jù)結(jié)構(gòu),稱為”位域”或”位段”。包含位域變量的結(jié)構(gòu)體叫作位域結(jié)構(gòu)體。位域結(jié)構(gòu)體的定義形式:

struct 位域結(jié)構(gòu)體名
{ 
類型說明符 位域名:位域長(zhǎng)度;
...
};

注意,位域長(zhǎng)度不應(yīng)該大于該類型說明符對(duì)應(yīng)的數(shù)據(jù)類型的位長(zhǎng)度。
使用位域的主要目的是壓縮存儲(chǔ),其大致規(guī)則為:

(1)如果相鄰位域字段的類型相同,且其位寬之和小于類型的sizeof大小,則后面的字段將緊鄰前一個(gè)字段存儲(chǔ),直到不能容納為止;
(2)如果相鄰位域字段的類型相同,但其位寬之和大于類型的sizeof大小,則后面的字段將從新的存儲(chǔ)單元開始,其偏移量為其類型大小的整數(shù)倍;
(3)如果相鄰位域字段的類型不同,則各編譯器的具體實(shí)現(xiàn)有差異,VC++采取不壓縮方式,GNU C++采取壓縮方式;
(4)如果位域字段之間穿插著非位域字段,則不進(jìn)行壓縮;
(5)整個(gè)結(jié)構(gòu)體的總大小為最寬基本類型成員大小的整數(shù)倍;
(6)位域可以無位域名,這時(shí)它只用作填充或調(diào)整位置,不能使用。例如:

struct BitFiledStruct
{ 
int a:1;
int :2; //該2位不能使用
int b:3;
int c:2;
};

關(guān)于位域結(jié)構(gòu)體的sizeof大小,考察如下代碼:

#include <iostream>
using namespace std;

struct BFS1
{
char f1 : 3;
char f2 : 4;
char f3 : 5;
};
struct BFS2
{
char f1 : 3;
int i : 4;
char f2 : 5;
};
struct BFS3
{
char f1 : 3;
char f2;
char f3 : 5;
};

int main()
{
cout<<sizeof(BFS1)<<endl;
cout<<sizeof(BFS2)<<endl;
cout<<sizeof(BFS3)<<endl;
}

運(yùn)行上面的程序,VC++和GNU C++輸出結(jié)果如下:

//VC++輸出結(jié)果
2
12
3

//GNU C++輸出結(jié)果
2
4
3

考察以上代碼,得出:

(1)sizeof(BFS1)==2。當(dāng)相鄰位域類型不同,在VC++中sizeof(BFS2)=1+pad(3)+4+1+pad(3)=12,采用不壓縮方式,位域變量i的偏移量需要是4的倍數(shù),并且位域結(jié)構(gòu)體BFS2的總大小必須是sizeof(int)的整數(shù)倍。在GNU C++中為sizeof(BFS2)=4,相鄰的位域字段的類型不同時(shí),采取了壓縮存儲(chǔ),位域變量i緊隨位域變量f1的剩余位進(jìn)行存儲(chǔ),位域變量f2同樣是緊隨位域變量i的剩余位進(jìn)行存儲(chǔ),并且位域結(jié)構(gòu)體BFS2的總大小必須是sizeof(int)的整數(shù)倍,所以最終結(jié)果sizeof(BFS2)=1+pad(3)=4。

(2)sizeof(BFS3)==3,當(dāng)非位域字段穿插在其中,不會(huì)產(chǎn)生壓縮,在VC++和GNU C++中得到的大小均為3,如果壓縮存儲(chǔ),則sizeof(BFS3)==2。

2.sizeof計(jì)算共用體

結(jié)構(gòu)體在內(nèi)存組織上是順序式的,共用體則是重疊式,各成員共享一段內(nèi)存,所以整個(gè)共用體的sizeof也就是每個(gè)成員sizeof的最大值。結(jié)構(gòu)體的成員也可以是構(gòu)造類型,這里,構(gòu)造類型成員是被作為整體考慮的。所以,下面例子中,假設(shè)sizeof(s)的值大于sizeof(i)和sizeof(c),那么sizeof(U)等于sizeof(s)。

union U
{
int i;
char c;
S1 s;
};

3.sizeof計(jì)算類

類是C++中常用的自定義構(gòu)造類型,有數(shù)據(jù)成員和成員函數(shù)組成,進(jìn)行sizeof計(jì)算時(shí),和結(jié)構(gòu)體并沒有太大的區(qū)別??疾烊缦麓a:

#include <iostream>
using namespace std;

class Small{};

class LessFunc
{
int num;
void func1(){};
};

class MoreFunc
{
int num;
void func1(){};
int func2(){return 1;};
};

class NeedAlign
{
char c;
double d;
int i;
};

class Virtual
{
int num;
virtual void func(){};
};

int main(int argc,char* argv[])
{
cout<<sizeof(Small)<<endl; //輸出1
cout<<sizeof(LessFunc)<<endl;//輸出4
cout<<sizeof(MoreFunc)<<endl;//輸出4
cout<<sizeof(NeedAlign)<<endl;//輸出24
cout<<sizeof(Virtual)<<endl; //輸出8
return 0;
}

注意一點(diǎn),C++中類同結(jié)構(gòu)體沒有本質(zhì)的區(qū)別,結(jié)構(gòu)體同樣可以包含成員函數(shù),構(gòu)造函數(shù),析構(gòu)函數(shù),虛函數(shù)和繼承,但一般不這么使用,沿用了C的結(jié)構(gòu)體使用習(xí)慣。類與結(jié)構(gòu)體唯一的區(qū)別就是結(jié)構(gòu)體的成員的默認(rèn)權(quán)限是public,而類是private。

基于以上這點(diǎn),再考察從程序的輸出結(jié)果,得出如下結(jié)論:

(1)類同結(jié)構(gòu)體一樣,C++中不允許長(zhǎng)度為0的數(shù)據(jù)類型存在,雖然類無任何成員,但該類的對(duì)象仍然占用1個(gè)字節(jié)。
(2)類的成員函數(shù)并不影響類對(duì)象占用的空間,類對(duì)象的大小是由它數(shù)據(jù)成員決定的。
(3)類和結(jié)構(gòu)體一樣,同樣需要對(duì)齊,具體對(duì)齊的規(guī)則見上文結(jié)構(gòu)體的內(nèi)存對(duì)齊。
(4)類如果包含虛函數(shù),編譯器會(huì)在類對(duì)象中插入一個(gè)指向虛函數(shù)表的指針,以幫助實(shí)現(xiàn)虛函數(shù)的動(dòng)態(tài)調(diào)用。

所以,該類的對(duì)象的大小至少比不包含虛函數(shù)時(shí)多4個(gè)字節(jié)。如果考慮內(nèi)存對(duì)齊,可能還要多些。如果使用數(shù)據(jù)成員之間的對(duì)齊,當(dāng)類對(duì)象至少包含一個(gè)數(shù)據(jù)成員,且擁有虛函數(shù),那么該對(duì)象的大小至少是8B,讀者可自行推導(dǎo)。

感謝各位的閱讀!關(guān)于C++中sizeof的作用是什么就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,讓大家可以學(xué)到更多知識(shí)。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到吧!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI