1. 程式人生 > >c++ 空類大小不為0的原因

c++ 空類大小不為0的原因

<span style="font-size:18px;">#include<iostream>
using namespace std;

class a{};

class b{};

class c:public a{
	virtual void fun()=0;
};

class d:public b,public c{
};

int main(){
	cout<<sizeof(a)<<endl;
	cout<<sizeof(b)<<endl;
	cout<<sizeof(c)<<endl;
	cout<<sizeof(d)<<endl;
	return 0;
}</span>

此例VC2010執行結果為1   1    4    8;codeblocks執行結果為1   1    4    4

為什麼會出現這種結果呢?類a,b明明是空類,它的大小應該為0,為什麼編譯器輸出的結果為1呢?這就是我們所要說的例項化的原因(空類同樣可以被例項化),每個例項在記憶體中都有一個獨一無二的地址,為了達到這個目的,編譯器往往會給一個空類隱含的加一個位元組,這樣空類在例項化後在記憶體得到了獨一無二的地址.所以a,b的大小為1.

類c是由類a派生而來,它裡面有一個純虛擬函式,由於有虛擬函式的原因,有一個指向虛擬函式的指標(vptr),在32位的系統分配給指標的大小為4個位元組,所以最後得到c類的大小為4.

類d是由類b,c派生而來,它的大小應該為二者之和5,為什麼卻是8或者4呢?這是因為為了提高例項在記憶體中的存取效率,類的大小往往被調整到系統的整數倍,並採取就近的法則,離哪個最近的倍數,就是該類的大小,所以類d的大小為8個位元組.當然在不同的編譯器上得到的結果可能不同(此處因為codeblocks自身有記憶體優化的機制)。但是這個實驗告訴我們,不管類是否為空類,均可被例項化(空類也可被例項化),每個被例項都有一個獨一無二的地址,每個非純虛類都能被例項化(即使是空類),類的例項化包括兩個過程,其一:new sizeof(類);其二,呼叫類的建構函式。所以類的大小必須大於0,才能返回一個有效地址。


#include<iostream>
using namespace std;

class a{
private:
    int data;
};

class b{
private:
    int data;
    static int Data;
};

int b::Data=0;
int main(int  argc,char* argv[]){
    cout<<sizeof(a)<<endl;
    cout<<sizeof(b)<<endl;
}
此例VC2010執行結果為4    4;codeblocks執行結果為4    4

為什麼類b多了一個數據成員卻和類a的大小相同呢?因為類b的靜態資料成員被編譯器放在程式的一個全域性靜態儲存區,它是類的一個數據成員.但是它不影響類的大小,不管這個類實際產生了多少例項,還是派生了多少新的類,靜態成員資料在類中永遠只有一個實體存在,而類的非靜態資料成員只有被例項化的時候,他們才存在.但是類的靜態資料成員一旦被宣告,無論類是否被例項化,它都已存在.可以這麼說,類的靜態資料成員是一種特殊的全域性變數,所以a,b的大小相同.

#include<iostream>
using namespace std;

class a{
private:
    int data;
    int Data;
public:
    a(int key){data=key;}
    ~a(){}
    void fun(int x){
        cout<<x<<endl;
    }
};

class b{
private:
    int data;
    int Data;
    static int data1;
};

int b::data1=0;
int main(int  argc,char* argv[]){
    a A(10);
    A.fun(5);
    cout<<sizeof(a)<<endl;
    cout<<sizeof(b)<<endl;
}
此例VC2010執行結果為5  8  8;codeblocks執行結果為5  8  8

它們的結果均相同,可以看出類的大小與它當中的建構函式,解構函式,以及其他的成員函式無關,只與它當中的成員資料有關.

從以上的幾個例子不難發現類的大小:

1、為類的非靜態成員資料的型別大小之和.

2、有編譯器額外加入的成員變數的大小,用來支援語言的某些特性(如:指向虛擬函式的指標).

3、為了優化存取效率,進行的邊緣調整.

4、與類中的建構函式,解構函式以及其他的成員函式無關.


轉載自:http://tc.uc.cn/?src=http%3A%2F%2Fhi.baidu.com%2Ftangsu2009%2Fitem%2Fc607a91646d15f4ce75e06f0&restype=1&ucshare=1&ucshareplatform=-1&country=cn#userconsent#



相關推薦

c++ 大小0原因

<span style="font-size:18px;">#include<iostream> using namespace std; class a{}; class b{}; class c:public a{ virtual voi

c++例項大小0原因收藏

初學者在學習面向物件的程式設計語言時,或多或少的都些疑問,我們寫的程式碼與最終生編譯成的程式碼卻 大相徑庭,我們並不知道編譯器在後臺做了什麼工作.這些都是由於我們僅停留在語言層的原因,所謂語言層就是教會我們一些基本的語法法則,但不會告訴我們為什麼這麼做?今天和大家談的一點感

c++例項大小0原因)

初學者在學習面向物件的程式設計語言時,或多或少的都些疑問,我們寫的程式碼與最終生編譯成的程式碼卻 大相徑庭,我們並不知道編譯器在後臺做了什麼工作.這 些都是由於我們僅停留在語言層的原因,所謂語言層就是教會我們一些基本的語法法則,但不會告訴我們為什麼這麼做?今天和大家談的一點感

c++例項大小0原因

初學者在學習面向物件的程式設計語言時,或多或少的都些疑問,我們寫的程式碼與最終生編譯成的程式碼卻 大相徑庭,我們並不知道編譯器在後臺做了什麼工作.這些都是由於我們僅停留在語言層的原因,所謂語言層就是教會我們一些基本的語法法則,但不會告訴我們為什麼這麼做?今天和大家談的一點感

[c#]基包含0個引數的建構函式

由於好久沒摸程式碼了 最近忙著找工作又重新寫點基礎的程式碼以備不時之需 今天在寫程式碼寫到關於派生類的問題上就突然出現瞭如標題的問題 一開始我是這樣寫的 然後就報了這個問題,我大概查了點資料‘ 大概的問題就出在Adult類的建構函式這裡 其實這裡的建構函式如果引數後面不寫

C二維陣列行為,列

二維陣列: 處理二維陣列得函式有一處可能不太容易理解:陣列的行可以在函式呼叫時傳遞,但是陣列的列卻只能被預置在函式內部。 eg: 1 #define COLS 4 2 int sum(int ar[][COLS],int rows){ 3 int r ; 4 int c;

C二維數組行為,列

col 兩個 int row font 就是 聲明 bsp 二維數組 二維數組: 處理二維數組得函數有一處可能不太容易理解:數組的行可以在函數調用時傳遞,但是數組的列卻只能被預置在函數內部。 eg: 1 #define COLS 4 2 int sum(int ar[

C++ 大小

class Base { }; int main(void) { cout  <<  "sizeof(Base) = " << sizeof(Base) << endl; return 1; } 程式輸出結果為 sizeof(Ba

C++

c++ tor emp 運算符 per 析構 ++ pan 有一個 class Empty { public: Empty(); // 缺省構造函數 Empty( const Empty& ); // 拷貝構造函數 ~

Java 數組 要求將以上數組中的 0 項去掉,將 0 的值存入一個新的數組

pac 數組 system 靜態 args 將不 pub bsp clas package com.xuyigang1234.chp01; public class Demo9 { public static void main(String[] args) {

select()返回0,但是read()返回0個位元組??

我有一段程式碼這樣寫的: char buf[maxMessageSize]; int maxfdp1; fd_set rset; FD_ZERO(&rset); //setbuf(stdout,NULL); while(1){

階乘問題(求 階乘最右邊 0 的數)

 P1134 階乘問題 4.9K通過 16.7K提交 題目提供者 評測方式雲端評測 標籤USACO高效能 難度普及/提高- 時空限制1000ms / 128MB  提交  

java檢測集合是否或長度是否0

java檢測集合是否為空或長度是否為0 當讀資料庫的資料後,返回的資料可能為空或長度等於0,而我們要取第一個,或其他的資料,如果這個資料為空時就會報錯,寫了一個工具類方便呼叫: package com.yx.utils; import org.junit.Test; import jav

StringUtils.join,原集合,結果

import org.apache.commons.collections.CollectionUtils; import org.apache.commons.lang.StringUtils;

java 中的指標,,的理解

一、null是代表不確定的物件 Java中,null是一個關鍵字,用來標識一個不確定的物件。因此可以將null賦給引用型別變數,但不可以將null賦給基本型別變數。 比如:int a = null;是錯誤的。Ojbect o = null是正確的。 Java中,變數的

c++-預設產生的成員

空類預設產生的成員: (1)預設建構函式(如果沒有定義其他的建構函式) (2)拷貝建構函式 (3)=運算子 (4)取地址運算子 (5)取地址運算子 const (6)解構函式 這裡只看一下(4)和(5) #include <iostream> using na

【初級】C++中子物件轉型物件時,編譯器默默做了什麼?

被問及C++中將子類物件轉型為父類物件時,會發生什麼?大部分人可能會回答說:子類成份會被"閹割",也就是說它不再具有子類特性。確實是這樣,不過我想知道編譯器是怎麼處理“閹割”的。請看如下程式碼, Child chd; Base b0(chd); Base b1 = chd

C++父強制轉換用法

-----Base.h #ifndef _BASE_H_ #define _BASE_H_ #include<iostream> using namespace std; class subclass; class base { public: int a;

sonarqube6程式碼覆蓋率顯示 單元測試

搭建完sonarqube6.1之後,按照網上找到的命令執行:mvn clean install  mvn sonar:sonar,開啟sonarqube網頁檢視結果時卻發現一個奇怪的現象: 覆蓋率

Can Place Flowers:陣列相鄰元素0的插入問題

Suppose you have a long flowerbed in which some of the plots are planted and some are not. However, flowers cannot be planted in adjace