欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解C++ sizeof(下)

 更新時(shí)間:2020年08月15日 10:17:33   作者:Dabelv  
這篇文章主要介紹了C++ sizeof的相關(guān)資料,幫助大家更好的理解和學(xué)習(xí)c++,感興趣的朋友可以了解下

sizeof作用于基本數(shù)據(jù)類型,在特定的平臺和特定的編譯器中,結(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)行成員變量的對齊?!队?jì)算機(jī)組成原理》一書中說明,對齊的目的是減少訪存指令周期,提高CPU存儲速度。

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

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

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

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

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

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中的對齊也遵守前三個準(zhǔn)則,因此sizeof(S2)=sizeof(char)+pad(3)+sizeof(S1)+1+pad(3)=1+3+8+1+3=16字節(jié),其中pad(3)表示填充3個字節(jié)。

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

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

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

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

1.2修改對齊方式

1.2.1#pragma pack

#pragma pack(n)中n為字節(jié)對齊數(shù),其取值為1、2、4、8、16,默認(rèn)是8。結(jié)構(gòu)體對齊時(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不能指定變量的存儲地址,變量的首地址默認(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ù)次冪,作用主要有兩個方面:
(1)使結(jié)構(gòu)體或類成員按#pragma pack確定內(nèi)存布局之后,在末尾填充內(nèi)存使得整個對象的大小至少是#的整數(shù)倍。
(2)作用于變量時(shí),強(qiáng)制要求編譯器將變量放置在地址是#整數(shù)倍的內(nèi)存位置上。這點(diǎn)在調(diào)用原生API等要求嚴(yán)格對齊的方法時(shí)十分重要。

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

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

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

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

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

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

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

(1)如果相鄰位域字段的類型相同,且其位寬之和小于類型的sizeof大小,則后面的字段將緊鄰前一個字段存儲,直到不能容納為止;
(2)如果相鄰位域字段的類型相同,但其位寬之和大于類型的sizeof大小,則后面的字段將從新的存儲單元開始,其偏移量為其類型大小的整數(shù)倍;
(3)如果相鄰位域字段的類型不同,則各編譯器的具體實(shí)現(xiàn)有差異,VC++采取不壓縮方式,GNU C++采取壓縮方式;
(4)如果位域字段之間穿插著非位域字段,則不進(jìn)行壓縮;
(5)整個結(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í),采取了壓縮存儲,位域變量i緊隨位域變量f1的剩余位進(jìn)行存儲,位域變量f2同樣是緊隨位域變量i的剩余位進(jìn)行存儲,并且位域結(jié)構(gòu)體BFS2的總大小必須是sizeof(int)的整數(shù)倍,所以最終結(jié)果sizeof(BFS2)=1+pad(3)=4。

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

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

結(jié)構(gòu)體在內(nèi)存組織上是順序式的,共用體則是重疊式,各成員共享一段內(nèi)存,所以整個共用體的sizeof也就是每個成員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++中不允許長度為0的數(shù)據(jù)類型存在,雖然類無任何成員,但該類的對象仍然占用1個字節(jié)。
(2)類的成員函數(shù)并不影響類對象占用的空間,類對象的大小是由它數(shù)據(jù)成員決定的。
(3)類和結(jié)構(gòu)體一樣,同樣需要對齊,具體對齊的規(guī)則見上文結(jié)構(gòu)體的內(nèi)存對齊。
(4)類如果包含虛函數(shù),編譯器會在類對象中插入一個指向虛函數(shù)表的指針,以幫助實(shí)現(xiàn)虛函數(shù)的動態(tài)調(diào)用。

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

以上就是詳解C++ sizeof(下)的詳細(xì)內(nèi)容,更多關(guān)于C++ sizeof的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 關(guān)于C++運(yùn)算符重載的一些困惑詳解

    關(guān)于C++運(yùn)算符重載的一些困惑詳解

    這篇文章主要給大家介紹了關(guān)于C++運(yùn)算符重載的一些困惑,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-04-04
  • Matlab實(shí)現(xiàn)三維投影繪制的示例代碼

    Matlab實(shí)現(xiàn)三維投影繪制的示例代碼

    這篇文章系小編為大家?guī)砹艘粋€三維投影繪制函數(shù)(三視圖繪制),函數(shù)支持三維曲線、曲面、三維多邊形、參數(shù)方程曲線、參數(shù)方程曲面的投影繪制,需要的可以參考一下
    2022-08-08
  • C++中static和const的作用和用法

    C++中static和const的作用和用法

    在C++中,"static"和"const"都是用于修飾變量的關(guān)鍵字,它們具有不同的作用,本文通過代碼示例給大家詳細(xì)的介紹static和const的作用和用法,,需要的朋友可以參考下
    2023-06-06
  • C語言實(shí)現(xiàn)詞法分析器

    C語言實(shí)現(xiàn)詞法分析器

    這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)詞法分析器,一個簡單的詞法分析程序,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-01-01
  • C語言結(jié)構(gòu)體內(nèi)存的對齊知識詳解

    C語言結(jié)構(gòu)體內(nèi)存的對齊知識詳解

    這篇文章主要介紹了C語言結(jié)構(gòu)體內(nèi)存的對齊的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-03-03
  • c++項(xiàng)目中后綴名vcxproj和sln的區(qū)別及說明

    c++項(xiàng)目中后綴名vcxproj和sln的區(qū)別及說明

    這篇文章主要介紹了c++項(xiàng)目中后綴名vcxproj和sln的區(qū)別及說明,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-05-05
  • C++使用Kruskal和Prim算法實(shí)現(xiàn)最小生成樹

    C++使用Kruskal和Prim算法實(shí)現(xiàn)最小生成樹

    這篇文章主要介紹了C++使用Kruskal和Prim算法實(shí)現(xiàn)最小生成樹,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-01-01
  • C++實(shí)現(xiàn)LeetCode(186.翻轉(zhuǎn)字符串中的單詞之二)

    C++實(shí)現(xiàn)LeetCode(186.翻轉(zhuǎn)字符串中的單詞之二)

    這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(186.翻轉(zhuǎn)字符串中的單詞之二),本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-08-08
  • 使用VC6.0對C語言程序進(jìn)行調(diào)試的基本手段分享

    使用VC6.0對C語言程序進(jìn)行調(diào)試的基本手段分享

    這篇文章主要介紹了用VC6.0開發(fā)c語言程序的時(shí)候調(diào)試代碼的一些小技巧,需要的朋友可以參考下
    2013-07-07
  • linux 匿名管道實(shí)例詳解

    linux 匿名管道實(shí)例詳解

    這篇文章主要介紹了linux 匿名管道實(shí)例詳解的相關(guān)資料,需要的朋友可以參考下
    2017-06-06

最新評論