1. 程式人生 > >c++空類例項大小不是0原因收藏

c++空類例項大小不是0原因收藏

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

首先:我們要知道什麼是類的例項化,所謂類的例項化就是在記憶體中分配一塊地址.

那我們先看看一個例子:

#include<iostream.h>

class a {};
class b{};
class c:public a{
virtual void fun()=0;
};
class d:public b,public c{};
int main()
{
cout<<"sizeof(a)"<<sizeof(a)<<endl;
cout<<"sizeof(b)"<<sizeof(b)<<endl;
cout<<"sizeof(c)"<<sizeof(c)<<endl;
cout<<"sizeof(d)"<<sizeof(d)<<endl;
return 0;}

程式執行的輸出結果為:

sizeof(a) =1

sizeof(b)=1

sizeof(c)=4

sizeof(d)=8

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

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

類d的大小更讓初學者疑惑吧,類d是由類b,c派生邇來的,它的大小應該為二者之和5,為什麼卻是8  呢?這是因為為了提高例項在記憶體中的存取效率.類的大小往往被調整到系統的整數倍.並採取就近的法則,裡哪個最近的倍數,就是該類的大小,所以類d的大小為8個位元組.

當然在不同的編譯器上得到的結果可能不同,但是這個實驗告訴我們初學者,不管類是否為空類,均可被例項化(空類也可被例項化),每個被例項都有一個獨一無二的地址.

我所用的編譯器為vc++ 6.0.

下面我們再看一個例子.

#include<iostream.h>
class a{
pivate:
int data;
};

class b{
private:
     int data;
static int data1;
};
int b::data1=0;
void mian(){
cout<<"sizeof(a)="<<sizeof(a)<<endl;
cout<<"sizeof(b)="<<sizeof(b)<<endl;
}

執行結果為:

sizeof(a)=4;

sizeof(b)=4;

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

所以a,b的大小相同.

下面我們看一個有建構函式,和解構函式的類的大小,它又是多大呢?

#include<iostream.h>
class A{
public :
A(int a){
   a=x;}
void f(int x){
   cout<<x<<endl;}
~A(){}

private:
   int x;
   int g;
   };
class B{
public:
private:
int data; int data2;
static int xs;
};
int B::xs=0;
void main(){
A s(10);
s.f(10);
cout<<"sozeof(a)"<<sizeof(A)<<endl;
cout<<"sizeof(b)"<<sizeof(B)<<endl;
}程式執行輸出結果為:

10 ,

sizeof(a) 8

sizeof(b) 8

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

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

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

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

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

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

相關推薦

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

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

c++例項大小0原因)

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

c++例項大小0原因

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

C++ 大小

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

為何大小是零?

如果上述程式碼中p1和p2相等,那麼說明編譯器作了優化。這樣的優化是安全的,而且非常有用。它允許程式設計師用空類來表示非常簡單的概念,而不需為此付出額外的(空間)代價。一些現代編譯器提供了這種“空基類優化”功能。

c++ 大小0原因

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

大小是多少?原因是什麼?

首先,我們要知道,一個類是沒有大小的,應該說成是類的例項的大小。 class A; A a; 其中,sizeof(A) == sizeof(a); 一個物件的大小 大於等於所有的非靜態成員大小的總和。 這是為什麼呢? 1.指向虛擬函式表的指標

C++

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

C++算術大小

div bsp bubuko sign .com pos class nbsp inf 1、int類型默認signed,最高位是符號位,表示範圍[-2^(n-1), 2^(n-1)-1],1byte=8bits 2、unsigned int,[0, 2^n-1] C++

c++-預設產生的成員

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

c++模板例項化 簡單例子

template<class T> 定義一個模板類注意模板類看作一個數據型別(int, float)class A { public: A(); T getlist(T x,int n); private: T data[maxsize]; }//構

linux下批量刪除檔案(大小等於0的檔案)的方法

程式碼如下   find . -name "*" -type f -size 0c | xargs -n 1 rm -f用這個還可以刪除指定大小的檔案,只要修改對應的 -size 引數就行,例如:程式碼如下        複製程式碼find . -name "*" -typ

C++的作用一

專案裡物件比較多,需要統一無效值,也需要根據該值判斷物件的狀態,nginx 開發指南使用空類來實現所有型別無效值的統一設定為-1 class NgxUnsetValue final {public:template<typename T>operator T

spring aop內部呼叫攔截原因及解決方案

       spring對應java web開發的同學來說,都不陌生,其中事務@Transactional在service層更是常常使用。 1.aop類內部呼叫不攔截原因 細心的同學也許早就發現當service中的某個沒標註@Transactional的方法呼叫另一個標註

為什麼C++中空結構體大小為1?

原文連結:http://www.spongeliu.com/260.html 對於結構體和空類大小是1這個問題,首先這是一個C++問題,在C語言下空結構體大小為0(當然這是編譯器相關的)。這裡的空類和空結構體是指類或結構體中沒有任何成員。 在C++下,空類和空結構體的大小是1(編譯器

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

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

C】sizeof(結構體/)的大小

sizeof(空類/空結構體) = 1; #include <iostream> using namespace std; // 空類 class ClassA { }; // 繼承空類的空類 class ClassB

c# 可

new section 如何 不清楚 erro row target 突出 不能 可空類型是 System.Nullable 結構的實例。可空類型可以表示其基礎值類型正常範圍內的值,再加上一個 null 值。例如,Nullable<Int32>,讀作&ldquo

C/C++中,數組、中空數組的解析及其作用

class 不同 理解 返回 free 而且 解析 分配 空類 轉自:http://blog.sina.com.cn/s/blog_93b45b0f01015s95.html 我們經常會遇到這些問題: (1)C++中定義一個空類,他們它的大小(sizeof) 為多少

學藝精而慚愧--論C++模板的使用

今天 lease stat entire 網上 永遠 hat efi 這樣的 自己斷斷續續地使用C++也有一段時間了。有些時候產生了自滿的情緒。覺得自己對C++的語言