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

基于C/C++將派生類賦值給基類的超詳細講解

 更新時間:2023年06月16日 15:50:17   作者:阿玥的小東東  
類其實也是一種數(shù)據(jù)類型,也可以發(fā)生數(shù)據(jù)類型轉(zhuǎn)換,下面這篇文章主要給大家介紹了關(guān)于基于C/C++將派生類賦值給基類的超詳細講解,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下

前言

在 C/C++中經(jīng)常會發(fā)生數(shù)據(jù)類型的轉(zhuǎn)換,例如將 int 類型的數(shù)據(jù)賦值給 float 類型的變量時,編譯器會先把 int 類型的數(shù)據(jù)轉(zhuǎn)換為 float 類型再賦值;反過來,float 類型的數(shù)據(jù)在經(jīng)過類型轉(zhuǎn)換后也可以賦值給 int 類型的變量。

數(shù)據(jù)類型轉(zhuǎn)換的前提是,編譯器知道如何對數(shù)據(jù)進行取舍。例如:

int a = 10.9;
printf("%d\n", a);

 輸出結(jié)果為 10,編譯器會將小數(shù)部分直接丟掉(不是四舍五入)。再如:

float b = 10;
printf("%f\n", b);

輸出結(jié)果為 10.000000,編譯器會自動添加小數(shù)部分。

類其實也是一種數(shù)據(jù)類型,也可以發(fā)生數(shù)據(jù)類型轉(zhuǎn)換,不過這種轉(zhuǎn)換只有在基類和派生類之間才有意義,并且只能將派生類賦值給基類,包括將派生類對象賦值給基類對象、將派生類指針賦值給基類指針、將派生類引用賦值給基類引用,這在 C++ 中稱為向上轉(zhuǎn)型(Upcasting)。相應地,將基類賦值給派生類稱為向下轉(zhuǎn)型(Downcasting)。

向上轉(zhuǎn)型非常安全,可以由編譯器自動完成;向下轉(zhuǎn)型有風險,需要程序員手動干預。本節(jié)只介紹向上轉(zhuǎn)型,向下轉(zhuǎn)型將在后續(xù)章節(jié)介紹。

向上轉(zhuǎn)型和向下轉(zhuǎn)型是面向?qū)ο缶幊痰囊环N通用概念,它們也存在于 Java、C#等編程語言中。

將派生類對象賦值給基類對象

下面的例子演示了如何將派生類對象賦值給基類對象:

#include <iostream>
using namespace std;
 
//基類
class A{
public:
    A(int a);
public:
    void display();
public:
    int m_a;
};
A::A(int a): m_a(a){ }
void A::display(){
    cout<<"Class A: m_a="<<m_a<<endl;
}
 
//派生類
class B: public A{
public:
    B(int a, int b);
public:
    void display();
public:
    int m_b;
};
B::B(int a, int b): A(a), m_b(b){ }
void B::display(){
    cout<<"Class B: m_a="<<m_a<<", m_b="<<m_b<<endl;
}
 
int main(){
    A a(10);
    B b(66, 99);
    //賦值前
    a.display();
    b.display();
    cout<<"--------------"<<endl;
    //賦值后
    a = b;
    a.display();
    b.display();
 
    return 0;
}

運行結(jié)果:

Class A: m_a=10
Class B: m_a=66, m_b=99
----------------------------
Class A: m_a=66
Class B: m_a=66, m_b=99

本例中 A 是基類, B 是派生類,a、b 分別是它們的對象,由于派生類 B 包含了從基類 A 繼承來的成員,因此可以將派生類對象 b 賦值給基類對象 a。通過運行結(jié)果也可以發(fā)現(xiàn),賦值后 a 所包含的成員變量的值已經(jīng)發(fā)生了變化。

賦值的本質(zhì)是將現(xiàn)有的數(shù)據(jù)寫入已分配好的內(nèi)存中,對象的內(nèi)存只包含了成員變量,所以對象之間的賦值是成員變量的賦值,成員函數(shù)不存在賦值問題。運行結(jié)果也有力地證明了這一點,雖然有a=b;這樣的賦值過程,但是 a.display() 始終調(diào)用的都是 A 類的 display() 函數(shù)。換句話說,對象之間的賦值不會影響成員函數(shù),也不會影響 this 指針。

將派生類對象賦值給基類對象時,會舍棄派生類新增的成員,也就是“大材小用”,如下圖所示:

 可以發(fā)現(xiàn),即使將派生類對象賦值給基類對象,基類對象也不會包含派生類的成員,所以依然不同通過基類對象來訪問派生類的成員。對于上面的例子,a.m_a 是正確的,但 a.m_b 就是錯誤的,因為 a 不包含成員 m_b。

這種轉(zhuǎn)換關(guān)系是不可逆的,只能用派生類對象給基類對象賦值,而不能用基類對象給派生類對象賦值。理由很簡單,基類不包含派生類的成員變量,無法對派生類的成員變量賦值。同理,同一基類的不同派生類對象之間也不能賦值。

要理解這個問題,還得從賦值的本質(zhì)入手。賦值實際上是向內(nèi)存填充數(shù)據(jù),當數(shù)據(jù)較多時很好處理,舍棄即可;本例中將 b 賦值給 a 時(執(zhí)行a=b;語句),成員 m_b 是多余的,會被直接丟掉,所以不會發(fā)生賦值錯誤。但當數(shù)據(jù)較少時,問題就很棘手,編譯器不知道如何填充剩下的內(nèi)存;如果本例中有b= a;這樣的語句,編譯器就不知道該如何給變量 m_b 賦值,所以會發(fā)生錯誤。

將派生類指針賦值給基類指針

除了可以將派生類對象賦值給基類對象(對象變量之間的賦值),還可以將派生類指針賦值給基類指針(對象指針之間的賦值)。我們先來看一個多繼承的例子,繼承關(guān)系為:

#include <iostream>
using namespace std;
 
//基類A
class A{
public:
    A(int a);
public:
    void display();
protected:
    int m_a;
};
A::A(int a): m_a(a){ }
void A::display(){
    cout<<"Class A: m_a="<<m_a<<endl;
}
 
//中間派生類B
class B: public A{
public:
    B(int a, int b);
public:
    void display();
protected:
    int m_b;
};
B::B(int a, int b): A(a), m_b(b){ }
void B::display(){
    cout<<"Class B: m_a="<<m_a<<", m_b="<<m_b<<endl;
}
 
//基類C
class C{
public:
    C(int c);
public:
    void display();
protected:
    int m_c;
};
C::C(int c): m_c(c){ }
void C::display(){
    cout<<"Class C: m_c="<<m_c<<endl;
}
 
//最終派生類D
class D: public B, public C{
public:
    D(int a, int b, int c, int d);
public:
    void display();
private:
    int m_d;
};
D::D(int a, int b, int c, int d): B(a, b), C(c), m_d(d){ }
void D::display(){
    cout<<"Class D: m_a="<<m_a<<", m_b="<<m_b<<", m_c="<<m_c<<", m_d="<<m_d<<endl;
}
 
int main(){
    A *pa = new A(1);
    B *pb = new B(2, 20);
    C *pc = new C(3);
    D *pd = new D(4, 40, 400, 4000);
 
    pa = pd;
    pa -> display();
 
    pb = pd;
    pb -> display();
 
    pc = pd;
    pc -> display();
 
    cout<<"-----------------------"<<endl;
    cout<<"pa="<<pa<<endl;
    cout<<"pb="<<pb<<endl;
    cout<<"pc="<<pc<<endl;
    cout<<"pd="<<pd<<endl;
 
    return 0;
}

運行結(jié)果:
Class A: m_a=4
Class B: m_a=4, m_b=40
Class C: m_c=400
-----------------------
pa=0x9b17f8
pb=0x9b17f8
pc=0x9b1800
pd=0x9b17f8

本例中定義了多個對象指針,并嘗試將派生類指針賦值給基類指針。與對象變量之間的賦值不同的是,對象指針之間的賦值并沒有拷貝對象的成員,也沒有修改對象本身的數(shù)據(jù),僅僅是改變了指針的指向。

1) 通過基類指針訪問派生類的成員

請讀者先關(guān)注第 68 行代碼,我們將派生類指針 pd 賦值給了基類指針 pa,從運行結(jié)果可以看出,調(diào)用 display() 函數(shù)時雖然使用了派生類的成員變量,但是 display() 函數(shù)本身卻是基類的。也就是說,將派生類指針賦值給基類指針時,通過基類指針只能使用派生類的成員變量,但不能使用派生類的成員函數(shù),這看起來有點不倫不類,究竟是為什么呢?第 71、74 行代碼也是類似的情況。

pa 本來是基類 A 的指針,現(xiàn)在指向了派生類 D 的對象,這使得隱式指針 this 發(fā)生了變化,也指向了 D 類的對象,所以最終在 display() 內(nèi)部使用的是 D 類對象的成員變量,相信這一點不難理解。

編譯器雖然通過指針的指向來訪問成員變量,但是卻不通過指針的指向來訪問成員函數(shù):編譯器通過指針的類型來訪問成員函數(shù)。對于 pa,它的類型是 A,不管它指向哪個對象,使用的都是 A 類的成員函數(shù)

概括起來說就是:編譯器通過指針來訪問成員變量,指針指向哪個對象就使用哪個對象的數(shù)據(jù);編譯器通過指針的類型來訪問成員函數(shù),指針屬于哪個類的類型就使用哪個類的函數(shù)。

2) 賦值后值不一致的情況

本例中我們將最終派生類的指針 pd 分別賦值給了基類指針 pa、pb、pc,按理說它們的值應該相等,都指向同一塊內(nèi)存,但是運行結(jié)果卻有力地反駁了這種推論,只有 pa、pb、pd 三個指針的值相等,pc 的值比它們都大。也就是說,執(zhí)行pc = pd;語句后,pc 和 pd 的值并不相等。

將派生類引用賦值給基類引用

引用在本質(zhì)上是通過指針的方式實現(xiàn)的,既然基類的指針可以指向派生類的對象,那么我們就有理由推斷:基類的引用也可以指向派生類的對象,并且它的表現(xiàn)和指針是類似的。

修改上例中 main() 函數(shù)內(nèi)部的代碼,用引用取代指針:

int main(){
    D d(4, 40, 400, 4000);
    A &ra = d;
    B &rb = d;
    C &rc = d;
    ra.display();
    rb.display();
    rc.display();
    return 0;
}

 運行結(jié)果:
Class A: m_a=4
Class B: m_a=4, m_b=40
Class C: m_c=400

ra、rb、rc 是基類的引用,它們都引用了派生類對象 d,并調(diào)用了 display() 函數(shù),從運行結(jié)果可以發(fā)現(xiàn),雖然使用了派生類對象的成員變量,但是卻沒有使用派生類的成員函數(shù),這和指針的表現(xiàn)是一樣的。

引用和指針的表現(xiàn)之所以如此類似,是因為引用和指針并沒有本質(zhì)上的區(qū)別,引用僅僅是對指針進行了簡單封裝

最后需要注意的是,向上轉(zhuǎn)型后通過基類的對象、指針、引用只能訪問從基類繼承過去的成員(包括成員變量和成員函數(shù)),不能訪問派生類新增的成員。

總結(jié)

到此這篇關(guān)于基于C/C++將派生類賦值給基類的超詳細講解的文章就介紹到這了,更多相關(guān)C/C++將派生類賦值給基類內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C語言中關(guān)于庫函數(shù) qsort 快排的用法

    C語言中關(guān)于庫函數(shù) qsort 快排的用法

    快速排序Qsort是所有學習算法和數(shù)據(jù)結(jié)構(gòu)最基礎的一個部分,也是考試題和面試的一個小重點。本片文章帶你了解Qsort的詳細用法規(guī)則
    2021-09-09
  • 雙緩沖解決VC++繪圖時屏幕閃爍

    雙緩沖解決VC++繪圖時屏幕閃爍

    相信很多人在做圖形界面開發(fā)時,常常會遇到屏幕閃爍的情況,當然我也不例外,下面我們就來詳細探討下這個問題的解決辦法
    2015-08-08
  • C語言使用鏈表實現(xiàn)學生籍貫管理系統(tǒng)

    C語言使用鏈表實現(xiàn)學生籍貫管理系統(tǒng)

    這篇文章主要為大家詳細介紹了C語言使用鏈表實現(xiàn)學生籍貫管理系統(tǒng),文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-02-02
  • Microsoft Visual C++ 6.0開發(fā)環(huán)境搭建教程

    Microsoft Visual C++ 6.0開發(fā)環(huán)境搭建教程

    這篇文章主要為大家詳細介紹了Microsoft Visual C++ 6.0開發(fā)環(huán)境搭建教程,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-04-04
  • C++簡易通訊錄系統(tǒng)實現(xiàn)流程詳解

    C++簡易通訊錄系統(tǒng)實現(xiàn)流程詳解

    這篇文章主要為大家介紹了C語言簡易版通訊錄的具體實現(xiàn)流程,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-05-05
  • C++簡明分析講解引用與函數(shù)提高及重載

    C++簡明分析講解引用與函數(shù)提高及重載

    今天繼續(xù)開始對C++核心編程知識的分享與系統(tǒng)講解,第一,這里會提到“引用”方法傳參以及剖析引用的本質(zhì);第二,我們對函數(shù)來一個提高,相當于進階函數(shù)了,包括函數(shù)的默認值,簡單的提一下函數(shù)的占位參數(shù),函數(shù)重載以及注意事項,接下來上正文
    2022-05-05
  • 基于C語言實現(xiàn)創(chuàng)意多彩貪吃蛇游戲

    基于C語言實現(xiàn)創(chuàng)意多彩貪吃蛇游戲

    這篇文章主要介紹了如何利用C語言實現(xiàn)一個創(chuàng)意多彩貪吃蛇游戲,這是一個純C語言外加easyx庫的繪圖函數(shù)制作而成的有趣小游戲,無需引入額外資源,感興趣的可以動手嘗試一下
    2022-08-08
  • C++關(guān)于字符的接收與輸出操作示例

    C++關(guān)于字符的接收與輸出操作示例

    這篇文章主要介紹了C++關(guān)于字符的接收與輸出操作,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習吧
    2023-01-01
  • 帶你了解C++this指針的用法及其深究

    帶你了解C++this指針的用法及其深究

    這篇文章主要介紹了C++中this指針的用法,對初學者而言是非常重要的概念,必須加以熟練掌握,需要的朋友可以參考下,希望能給你帶來幫助
    2021-08-08
  • C語言Turbo C下實現(xiàn)俄羅斯方塊

    C語言Turbo C下實現(xiàn)俄羅斯方塊

    這篇文章主要為大家詳細介紹了C語言Turbo C下寫的俄羅斯方塊,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-02-02

最新評論