C#面向對象核心-繼承

来源:https://www.cnblogs.com/tanyuyang/archive/2023/03/28/17265952.html
-Advertisement-
Play Games

繼承 繼承主要實現重用代碼,來節省開發時間。 1 繼承基本概念 一個類B繼承一個類A,被繼承的類A稱為 父類、基類、超類,繼承的類B稱為 子類、派生類。 子類會繼承父類的所有成員 子類擁有父類的所有特征和行為 子類可以有自己的特征行為 C#中允許子類和父類存在同名的成員,但不建議使用 特點: 單根性 ...


繼承

繼承主要實現重用代碼,來節省開發時間。

1 繼承基本概念

一個類B繼承一個類A,被繼承的類A稱為 父類、基類、超類,繼承的類B稱為 子類、派生類。

  • 子類會繼承父類的所有成員
  • 子類擁有父類的所有特征和行為
  • 子類可以有自己的特征行為
  • C#中允許子類和父類存在同名的成員,但不建議使用

特點:

  • 單根性 子類只能有一個父類,不能多繼承
  • 傳遞性 子類可以間接繼承父類的父類

1.1 基本語法

class Teacher
{
    public string name;
    public int number;

    public void SpeakName()
    {
        Console.WriteLine(name);
    }
}

class TeachingTeacher : Teacher
{
    public string subject;//科目

    public void SpeakSubject()
    {
        Console.WriteLine($"我是{subject}老師");
    }
}

class ChineseTeacher : TeachingTeacher
{
    public void Skill()
    {
        Console.WriteLine("餘幼時即嗜學");
    }
}

1.2 使用

TeachingTeacher tt = new TeachingTeacher();
tt.name = "abc";
tt.number = 1;
tt.SpeakName();
tt.subject = "C#";
tt.SpeakSubject();

ChineseTeacher ct = new ChineseTeacher();
ct.name = "def";
ct.number = 2;
ct.SpeakName();
ct.subject = "Chinese";
ct.SpeakSubject();
ct.Skill();

2 里氏替換原則

里氏替換原則(Liskov Substitution principle):子類可以擴展父類的功能,但不能改變父類原有的功能,是面向對象七大原則中最重要的原則。
概念:任何父類出現的地方,子類都可以替代
重點:語法表現一父類容器裝子類對象,因為子類對象包含了父類的所有內容
作用:方便進行對象存儲和管理

2.1 基本語法

class GameObject
{

}

class Player : GameObject
{
    public void PlayerAttack()
    {
        Console.WriteLine("玩家攻擊");
    }
}

class Monster : GameObject
{
    public void MonsterAttack()
    {
        Console.WriteLine("怪物攻擊");
    }
}

class Boss : GameObject
{
    public void BossAttack()
    {
        Console.WriteLine("Boss攻擊");
    }
}

//Main
//里氏替換原則 用父類容器裝載子類對象
GameObject p = new Player();
GameObject m = new Monster();
GameObject b = new Boss();

2.2 is 和 as

is:判斷一個對象是否為指定對象,返回:bool,是為真,不是為假

as:將一個對象轉換為指定類對象,返回:成功返回指定類對象,失敗返回null

基本語法:

  • 類對象 is 類名,語句返回真或假
  • 類對象 as 類名,語句返回對象或null
if(p is Player)
{
    Player p1 = p as Player;
    p1.PlayerAttack();
}

3 繼承的構造函數

3.1 基本概念

子類聲明子類對象時,先執行父類的構造函數,再執行子類的構造函數。

  • 父類的無參構造很重要,有參構造函數會自動頂掉無參構造函數,若寫了有參構造,則最好還要寫無參構造
  • 子類可以通過base關鍵字代表父類調用父類構造函數
  • 構造函數執行順序:父類的父類的構造函數——父類的構造函數——子類的構造函數
class GameObject
{
    public GameObject()
    {
        Console.WriteLine("GameObject的構造函數");
    }
}
class Player : GameObject
{
    public Player()
    {
        Console.WriteLine("Player的構造函數");
    }
}
class MainPlayer : Player
{
    public MainPlayer()
    {
        Console.WriteLine("MainPlayer的構造函數");
    }
}

//Main
MainPlayer mp = new MainPlayer();
/*
輸出:
GameObject的構造函數
Player的構造函數
MainPlayer的構造函數
*/
Father構造
Son一個參數構造
Son兩個個參數構造

3.2 base關鍵字

通過 base(參數) 可以調用指定父類構造。

class Father
{
    /*public Father()
    {

    }*/
    public Father(int i)//有參構造函數會自動頂掉無參構造函數
    {
        Console.WriteLine("Father構造");
    }
}
//子類實例化時預設先調用父類的無參構造,如果父類無參構造被頂掉,就會報錯
class Son : Father
{
    //通過base改變預設調用父類的無參構造,指定為有參構造
    public Son(int i) : base(i)
    {
        Console.WriteLine("Son一個參數構造");
    }

    public Son(int i, string str) : this(i)//this(i)表示調用一個參數的構造函數public Son(int i) : base(i){}
    {
        Console.WriteLine("Son兩個個參數構造");
    }
}

//Main
Son s = new Son(1, "123");
/*
輸出:
Father構造
Son一個參數構造
Son兩個個參數構造
*/

4 object 和裝箱拆箱

4.1 object

object 是所有類型的父類,它是一個類(引用類型)。

作用:

  • 利用里氏替換原則,用object容器裝所有對象
  • 表示不確定類型,作為函數參數類型

4.2 使用

class Father
{

}

class Son : Father
{
    public void Speak()
    {

    }
}

//Main
Father f = new Son();//用父類容器裝載子類
if(f is Son)
{
    (f as Son).Speak();
}

//用object裝載萬物,然後轉換成所需類型來使用
//1、引用類型
object o = new Son();
//用is和as來判斷和轉換即可
if(o is Son)
{
    (o as Son).Speak();
}

//2、值類型
object o2 = 1f;
//用強轉
float fl = (float)o2;

//特殊string類型
object str = "123";
string str2 = str as string;

//數組
object arr = new int[10];
int[] ar = arr as int[];

4.3 裝箱和拆箱

  • 裝箱:用object來存值類型,即值類型轉換為object類型 ,棧記憶體會遷移到堆記憶體中
  • 拆箱:再把object轉為值類型 ,堆記憶體會遷移到棧記憶體中
  • 好處:不確定類型時可以方便參數的存儲和傳遞
  • 壞處:存在記憶體遷移,增加性能消耗
object v = 3;//裝箱
int a = (int)v;//拆箱

5 sealed 密封類

  • 使用 sealed 關鍵字修飾的類,讓類無法被繼承
  • 在面向對象程式的設計中,密封類的主要作用就是不允許最底層子類被繼承,可以保證程式的規範性、安全性
sealed class Father
{

}

class Son : Father//報錯,無法繼承
{

}

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、Jx9 虛擬機的生命周期 載入 Jx9 腳本 jx9_compile() 或 jx9_compile_file(),載入編譯成功後,Jx9 引擎將自動創建一個實例 (jx9_vm) 並且返回指向此虛擬機的指針用於後續調用。 如載入編譯 Jx9 腳本時出現問題,也就是編譯時出錯,可調用jx9_co ...
  • 生產環境,一個簡單的事務方法,提交失敗,報 Global lock wait timeout 偽代碼如下: @GlobalTransactional(rollbackFor = Exception.class,timeoutMills = 30000,lockRetryInternal=3000,l ...
  • PowerPlume是PowerBuilder深度創新的擴展開發框架(免費商用)。 它的三個主要特色是一、原創功能;二、零改動相容(非侵入性);三、極簡介面設計。 ...
  • 記錄人生第一次重裝系統之後的數據恢復過程,包括桌面恢復、常用軟體下載和屬性修改、vscode插件、zotero數據恢復、onenote筆記數據恢復,讓重裝系統的你不用慌。 ...
  • 發文原因 很多初學者都使用 cargo new [project_name] 來創建項目,並直接在 main.rs 文件中實現所有功能。 這樣是不合理的,並不符合我們 cargo 的開發規範。 下麵將簡單的介紹一下 rust project 中的文件結構。 cargo new [project_na ...
  • 其他 1 命名空間 命名空間用來組織和重用代碼的,命名空間就像一個工具包,類就像工具。 1.1 使用 namespace MyGame { class GameObject { } } namespace MyGame//命名空間可以分開寫 { class Player : GameObject { ...
  • 多態 1 認識多態 1.1 基本概念 多態是同一個行為具有多個不同表現形式或形態的能力,意味著有多重形式。在面向對象編程範式中,多態性往往表現為"一個介面,多個功能"。 在 C# 中,每個類型都是多態的,因為包括用戶定義類型在內的所有類型都繼承自 Object。 多態性分為靜態的和動態多態。在靜態多 ...
  • 微軟發佈 C# async/await 非同步語法功能已經好久了,但是目前來看使用並不廣泛。本人經過實踐在開發過程中使用 async/await 一路到底確實很爽,而且也沒有啥問題。但是在面對舊項目變更要使用些功能的時候可能會遇到同步方法調用非同步方法的情況,本人在這種情況就發生調用沒有響應的問題,並作 ...
一周排行
    -Advertisement-
    Play Games
  • ## 引言 最近發現自己喜歡用的 Todo 軟體總是差點意思,畢竟每個人的習慣和工作流不太一樣,我就想著自己寫一個小的[Todo 項目]( https://github.com/circler3/TodoTrack ),核心的功能是自動記錄 Todo 執行過程中消耗的時間(尤其面向程式員),按照自己 ...
  • ### 前言 當我們編寫 C# 代碼時,經常需要處理大量的數據集合。在傳統的方式中,我們往往需要先將整個數據集合載入到記憶體中,然後再進行操作。但是如果數據集合非常大,這種方式就會導致記憶體占用過高,甚至可能導致程式崩潰。 C# 中的`yield return`機制可以幫助我們解決這個問題。通過使用`y ...
  • 1. ADO.NET的前世今生 ADO.NET的名稱起源於ADO(ActiveX Data Objects),是一個COM組件庫,用於在以往的Microsoft技術中訪問數據。之所以使用ADO.NET名稱,是因為Microsoft希望表明,這是在NET編程環境中優先使用的數據訪問介面。 ADO.NE ...
  • 1. 為什麼需要單元測試 在我們之前,測試某些功能是否能夠正常運行時,我們都將代碼寫到Main方法中,當我們測試第二個功能時,我們只能選擇將之前的代碼清掉,重新編寫。此時,如果你還想重新測試你之前的功能時,這時你就顯得有些難為情了,因為代碼都被你清掉了。當然你完全可以把代碼寫到一個記事本中進行記錄, ...
  • 1. 透過現象看本質 反射被譽為是 c#中的黑科技 ,在很多領域中都有反射的身影,例如,我們經常使用的ORM框架,ABP框架 等。 反射指程式可以訪問、檢測和修改它本身狀態或行為的一種能力。. 程式集包含模塊,而模塊包含類型,類型又包含成員。. 反射則提供了封裝程式集、模塊和類型的對象。. 您可以使 ...
  • # Rust Web 全棧開發之 Web Service 中的錯誤處理 ## Web Service 中的統一錯誤處理 ### Actix Web Service 自定義錯誤類型 -> 自定義錯誤轉為 HTTP Response - 資料庫 - 資料庫錯誤 - 串列化 - serde 錯誤 - I/ ...
  • 在前面的幾篇文章中,詳細地給大家介紹了Java里的集合。但在介紹集合時,我們涉及到了泛型的概念卻並沒有詳細學習,所以今天我們要花點時間給大家專門講解什麼是泛型、泛型的作用、用法、特點等內容 ...
  • ###BIO:同步阻塞 主線程發起io請求後,需要等待當前io操作完成,才能繼續執行。 ###NIO:同步非阻塞 引入selector、channel、等概念,當主線程發起io請求後,輪詢的查看系統是否準備好執行io操作,沒有準備好則主線程不會阻塞會繼續執行,準備好主線程會阻塞等待io操作完成。 # ...
  • 摘要:在讀多寫少的環境中,有沒有一種比ReadWriteLock更快的鎖呢?有,那就是JDK1.8中新增的StampedLock! 本文分享自華為雲社區《【高併發】高併發場景下一種比讀寫鎖更快的鎖》,作者: 冰 河。 什麼是StampedLock? ReadWriteLock鎖允許多個線程同時讀取共 ...
  • ## 併發與並行😣 ### 併發與並行的概念和區別 並行:同一個時間段內多個任務同時在不同的CPU核心上執行。強調同一時刻多個任務之間的”**同時執行**“。 併發:同一個時間段內多個任務都在進展。強調多個任務間的”**交替執行**“。 ![](https://img2023.cnblogs.co ...