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
  • 概述:在C#中,++i和i++都是自增運算符,其中++i先增加值再返回,而i++先返回值再增加。應用場景根據需求選擇,首碼適合先增後用,尾碼適合先用後增。詳細示例提供清晰的代碼演示這兩者的操作時機和實際應用。 在C#中,++i 和 i++ 都是自增運算符,但它們在操作上有細微的差異,主要體現在操作的 ...
  • 上次發佈了:Taurus.MVC 性能壓力測試(ap 壓測 和 linux 下wrk 壓測):.NET Core 版本,今天計劃準備壓測一下 .NET 版本,來測試並記錄一下 Taurus.MVC 框架在 .NET 版本的性能,以便後續持續優化改進。 為了方便對比,本文章的電腦環境和測試思路,儘量和... ...
  • .NET WebAPI作為一種構建RESTful服務的強大工具,為開發者提供了便捷的方式來定義、處理HTTP請求並返迴響應。在設計API介面時,正確地接收和解析客戶端發送的數據至關重要。.NET WebAPI提供了一系列特性,如[FromRoute]、[FromQuery]和[FromBody],用 ...
  • 原因:我之所以想做這個項目,是因為在之前查找關於C#/WPF相關資料時,我發現講解圖像濾鏡的資源非常稀缺。此外,我註意到許多現有的開源庫主要基於CPU進行圖像渲染。這種方式在處理大量圖像時,會導致CPU的渲染負擔過重。因此,我將在下文中介紹如何通過GPU渲染來有效實現圖像的各種濾鏡效果。 生成的效果 ...
  • 引言 上一章我們介紹了在xUnit單元測試中用xUnit.DependencyInject來使用依賴註入,上一章我們的Sample.Repository倉儲層有一個批量註入的介面沒有做單元測試,今天用這個示例來演示一下如何用Bogus創建模擬數據 ,和 EFCore 的種子數據生成 Bogus 的優 ...
  • 一、前言 在自己的項目中,涉及到實時心率曲線的繪製,項目上的曲線繪製,一般很難找到能直接用的第三方庫,而且有些還是定製化的功能,所以還是自己繪製比較方便。很多人一聽到自己畫就害怕,感覺很難,今天就分享一個完整的實時心率數據繪製心率曲線圖的例子;之前的博客也分享給DrawingVisual繪製曲線的方 ...
  • 如果你在自定義的 Main 方法中直接使用 App 類並啟動應用程式,但發現 App.xaml 中定義的資源沒有被正確載入,那麼問題可能在於如何正確配置 App.xaml 與你的 App 類的交互。 確保 App.xaml 文件中的 x:Class 屬性正確指向你的 App 類。這樣,當你創建 Ap ...
  • 一:背景 1. 講故事 上個月有個朋友在微信上找到我,說他們的軟體在客戶那邊隔幾天就要崩潰一次,一直都沒有找到原因,讓我幫忙看下怎麼回事,確實工控類的軟體環境複雜難搞,朋友手上有一個崩潰的dump,剛好丟給我來分析一下。 二:WinDbg分析 1. 程式為什麼會崩潰 windbg 有一個厲害之處在於 ...
  • 前言 .NET生態中有許多依賴註入容器。在大多數情況下,微軟提供的內置容器在易用性和性能方面都非常優秀。外加ASP.NET Core預設使用內置容器,使用很方便。 但是筆者在使用中一直有一個頭疼的問題:服務工廠無法提供請求的服務類型相關的信息。這在一般情況下並沒有影響,但是內置容器支持註冊開放泛型服 ...
  • 一、前言 在項目開發過程中,DataGrid是經常使用到的一個數據展示控制項,而通常表格的最後一列是作為操作列存在,比如會有編輯、刪除等功能按鈕。但WPF的原始DataGrid中,預設只支持固定左側列,這跟大家習慣性操作列放最後不符,今天就來介紹一種簡單的方式實現固定右側列。(這裡的實現方式參考的大佬 ...