第十單元 面向對象二:靜態類與靜態成員

来源:https://www.cnblogs.com/xuyubing/archive/2023/05/30/17442995.html
-Advertisement-
Play Games

前言 我們天天都在使用 Console.WriteLine("Hello world"); ​ Console.ReadLine(""); ​ Arrays.Sort() 為什麼Console 類 調用 方法不需要實例化而可以直接調用呢? 我們可以查看Console的源碼發現Console類定義如下 ...


前言

我們天天都在使用

Console.WriteLine("Hello world");
​
Console.ReadLine("");
​
Arrays.Sort()

  

為什麼Console 類 調用 方法不需要實例化而可以直接調用呢?

我們可以查看Console的源碼發現Console類定義如下:

public static class Console
{
    // ... 
}

  

我們發現 Console 類 前面有個 static 關鍵字修飾, 我們把 static 修飾的類 叫作 靜態類

1. 靜態類

static關鍵字的作用

  • static是靜態的意思,可以修飾成員變數和成員方法。

  • static修飾成員變數表示該成員變數只在記憶體中只存儲一份,可以被共用訪問、修改。

public static class MyConsole 
{
    // 靜態構造方法
    static MyConsole()
    {

    }
    // 靜態變數
    public static string Title { get; set; }

    public static string Description {  get; set; }
    // 靜態方法
    public static void WriteLine(params string[] s)
    {

    }
}

class Program
{
    static void Main(string[] args)
    {
        MyConsole.Title = "Hello,靜態類";
        MyConsole.WriteLine("靜態方法調用");
    }
}

 

靜態類基本上與非靜態類相同,但存在一個差異:靜態類無法實例化。 換句話說,無法使用 new 運算符創建類類型的變數。 由於不存在任何實例變數,因此可以使用類名本身訪問靜態類的成員

與所有類類型的情況一樣,載入引用該類的程式時,.NET 運行時會載入靜態類的類型信息。 程式無法確切指定類載入的時間。 但是,可保證進行載入,以及在程式中首次引用類之前初始化其欄位並調用其靜態構造函數。 靜態構造函數只調用一次,在程式所駐留的應用程式域的生存期內,靜態類會保留在記憶體中。

 

以下列表提供靜態類的主要功能:

  • 只包含靜態成員。

  • 無法進行實例化。

  • 會進行密封。

  • 不能包含實例構造函數

因此,創建靜態類基本上與創建只包含靜態成員和私有構造函數的類相同。 私有構造函數可防止類進行實例化。 使用靜態類的優點是編譯器可以進行檢查,以確保不會意外地添加任何實例成員。 編譯器可保證無法創建此類的實例。

靜態類會進行密封,因此不能繼承。 它們不能繼承自任何類(除了 Object)。 靜態類不能包含實例構造函數。 但是,它們可以包含靜態構造函數。 如果非靜態類包含了需要進行有意義的初始化的靜態成員,則它也應該定義一個靜態構造器。

2. 非靜態類的靜態成員

非靜態類

不被static 關鍵字修飾的類稱為非靜態類.

 

非靜態類可以包含靜態方法、欄位、屬性或事件。 即使未創建類的任何實例,也可對類調用靜態成員。 靜態成員始終按類名(而不是實例名稱)進行訪問。 靜態成員只有一個副本存在(與創建的類的實例數無關)。 靜態方法和屬性無法在其包含類型中訪問非靜態欄位和事件,它們無法訪問任何對象的實例變數,除非在方法參數中顯式傳遞它。

更典型的做法是聲明具有一些靜態成員的非靜態類(而不是將整個類都聲明為靜態)。 靜態欄位的兩個常見用途是保留已實例化的對象數的計數,或是存儲必須在所有實例間共用的值。

靜態方法可以進行重載,但不能進行替代,因為它們屬於類,而不屬於類的任何實例。

雖然欄位不能聲明為 static const,不過 static const 欄位在其行為方面本質上是靜態的。 它屬於類型,而不屬於類型的實例。 因此,可以使用用於靜態欄位的相同 ClassName.MemberName 表示法來訪問 const 欄位。 無需進行對象實例化。

C# 不支持靜態局部變數(即在方法範圍中聲明的變數)。

可在成員的返回類型之前使用 static 關鍵字聲明靜態類成員,如下麵的示例所示:

public class Cat // 非靜態類
{
    public string Name { get; set; } // 實例變數,屬於對象
    public static int Leg { get; set; } // 靜態成員變數,, 屬於類

    // 重載靜態方法run, 屬於類
    public static void Run()
    {
        // Console.WriteLine(this.Name);// 報錯,靜態方法中,不能訪問實例成員
        Console.WriteLine($"{Leg} 條腿的貓在跑步");// 正確,靜態類中可以訪問靜態成員變數
    }
    // 重載靜態方法run, 屬於類
    public static void Run(int speed)
    {
        Console.WriteLine($"{Leg} 條腿的貓在跑步,速度是:{speed}");
    }
    
    // 實例方法,屬於對象
    public void Eat()
    {
        // 實例方法既可以訪問成員變數,也可以訪問靜態變數
        Console.WriteLine($"{Leg} 條腿的 {this.Name} 在搶狗糧吃。") ; 
    }
}

 

在首次訪問靜態成員之前以及在調用構造函數(如果有)之前,會初始化靜態成員。 若要訪問靜態類成員,請使用類的名稱(而不是變數名稱)指定成員的位置,如下麵的示例所示:

static void Main(string[] args)
{
    Cat.Leg = 4;
    Cat.Run(30);
    
    Cat c = new Cat();
    c.Eat();
}

 

3. 靜態構造方法

靜態構造函數用於初始化任何靜態數據,或執行僅需執行一次的特定操作。 將在創建第一個實例或引用任何靜態成員之前自動調用靜態構造函數。

class SimpleClass
{
    // Static variable that must be initialized at run time.
    static readonly long baseline;

    // Static constructor is called at most one time, before any
    // instance constructor is invoked or member is accessed.
    static SimpleClass()
    {
        baseline = DateTime.Now.Ticks;
    }
}

 

靜態構造函數具有以下屬性:

  • 靜態構造函數不使用訪問修飾符或不具有參數。

  • 類或結構只能有一個靜態構造函數。

  • 靜態構造函數不能繼承或重載。

  • 靜態構造函數不能直接調用,並且僅應由公共語言運行時 (CLR) 調用。 可以自動調用它們。

  • 用戶無法控制在程式中執行靜態構造函數的時間。

  • 自動調用靜態構造函數。 它在創建第一個實例或引用該類(不是其基類)中聲明的任何靜態成員之前初始化。 靜態構造函數在實例構造函數之前運行。 調用(而不是分配)分配給事件或委托的靜態方法時,將調用類型的靜態構造函數。 如果靜態構造函數類中存在靜態欄位變數初始值設定項,它們將以在類聲明中顯示的文本順序執行。 初始值設定項緊接著執行靜態構造函數之前運行。

  • 如果未提供靜態構造函數來初始化靜態欄位,會將所有靜態欄位初始化為其預設值,如 C# 類型的預設值中所列。

  • 如果靜態構造函數引發異常,運行時將不會再次調用該函數,並且類型在應用程式域的生存期內將保持未初始化。 大多數情況下,當靜態構造函數無法實例化一個類型時,或者當靜態構造函數中發生未經處理的異常時,將引發 TypeInitializationException 異常。 對於未在源代碼中顯式定義的靜態構造函數,故障排除可能需要檢查中間語言 (IL) 代碼。

  • 靜態構造函數的存在將防止添加 BeforeFieldInit 類型屬性。 這將限制運行時優化。

  • 聲明為 static readonly 的欄位可能僅被分配為其聲明的一部分或在靜態構造函數中。 如果不需要顯式靜態構造函數,請在聲明時初始化靜態欄位,而不是通過靜態構造函數,以實現更好的運行時優化。

  • 運行時在單個應用程式域中多次調用靜態構造函數。 該調用是基於特定類型的類在鎖定區域中進行的。 靜態構造函數的主體中不需要其他鎖定機制。 若要避免死鎖的風險,請勿阻止靜態構造函數和初始值設定項中的當前線程。 例如,不要等待任務、線程、等待句柄或事件,不要獲取鎖定,也不要執行阻止並行操作,如並行迴圈、Parallel.Invoke 和並行 LINQ 查詢。

單例模式實現

class MyDbContext
{
    private static MyDbContext _instance;
    // 禁止外部實例化
    private MyDbContext()
    {

    }

    public static string DbName;
    public static string UserName;
    public static string Password;
    
    // 靜態構造方法
    static MyDbContext()
    {
        DbName = "SqlServer";
        UserName = "sa";
        Password = "123456";
        // 實例化一次
        _instance = new MyDbContext();
    }
    // 對外提供訪問的路徑
    public MyDbContext Instance
    {
        get { return _instance; }
    }

    public void Connection()
    {

    }

    public void Abort()
    {

    }
}

 

4. 作業

  1. 定義一個靜態類 MyMath

  2. MyMath類中 實現兩個參數的方法:加,減,乘,除,平均數

  3. 重載 :加,減,乘,除,平均數

  4. 定義一個非靜態類 CommonUtils, 任意聲明兩個實例成員變數與靜態成員變數

  5. CommonUtils類中,聲明一個實例方法,並輸出 上述的實例成員變數與靜態成員變數。

  6. CommonUtils 類中, 聲明一個靜態方法,並輸出靜態成員變數。

  

視頻教程:

譽尚學教育_譽尚學教育騰訊課堂官網 (qq.com)
或者:
C# 最強入門編程(.Net 學習系列開山巨作)_嗶哩嗶哩_bilibili

海闊平魚躍,天高任我行,給我一片藍天,讓我自由翱翔。
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在最近的互聯網項目開發中,需要獲取用戶的訪問ip信息,併進行後續統計分析。 這些ip信息是在第三方的服務中分組存放的,且每個分組都都是分頁(1頁10條)存放的,如果一次性訪問大量的數據,API很有可能會報錯。 怎樣通過HTTP的方式去獲取到信息,並且模擬瀏覽器每頁每頁獲取10條的信息,且持久到資料庫... ...
  • 如前所述,在前幾章內容中筆者簡單介紹了`記憶體讀寫`的基本實現方式,這其中包括了`CR3切換`讀寫,`MDL映射`讀寫,`記憶體拷貝`讀寫,本章將在如前所述的讀寫函數進一步封裝,並以此來實現驅動讀寫記憶體浮點數的目的。記憶體`浮點數`的讀寫依賴於`讀寫記憶體位元組`的實現,因為浮點數本質上也可以看作是一個位元組集... ...
  • # 還在愁個人博客沒有圖片放?🥱 # 前言 之前將爬取的圖片下載後,我也是放到我了我的博客上面 [ZY知識庫 · ZY - Home Page (pljzy.top)](https://pljzy.top/) 然後順便寫了一個隨機返回圖片的介面,現在我就來說一下如何使用這個介面,以便在自己的博客上 ...
  • 在 WPF 中,CanContentScroll 是 ScrollViewer 控制項的一個附加屬性,它控制滾動視圖中的內容是否按項或像素來滾動。 當 CanContentScroll 設置為 false 時,表示 ScrollViewer 控制項使用逐像素的滾動方式,這意味著滾動視圖中的內容會以像素為 ...
  • 1. 泛型 泛指某種類型。 1、使用參數形式定義 2、使用時傳入具體類型 3、編譯時檢查類型安全 4、邏輯上是多個不同類型 泛型與非泛型之間的區別 性能高:可以避免裝箱和拆箱操作 類型安全 :在進行類型轉換的時候不會拋出異常 代碼重用:定義一次,用許多種不同類型實例化 代碼擴展性好 ArrayLis ...
  • 編寫包含多個 `csproj` 的程式時,隨著項目數量的持續增加,可能涉及一些文件夾的變動,手動添加項目或者變動會變得非常麻煩,這個時候,可以利用 `dotnet cli` 幫助我們完成。 如果從零開始,我們可以新建一個解決方案。 ```powershell dotnet new sln -n to ...
  • # 依賴註入的使用 ## 構造方法註入 這是將服務註入類的最常用方法,是將依賴項註入類的首選方式,也是微軟推崇的模式。這樣,除非提供了所有構造方法註入的依賴項,否則無法構造類,顯示的聲明瞭類必需的服務,使開發人員一目瞭然。 ```csharp public class BookAppService ...
  • 假設老師類設計如下: class 老師類 { 屬性:姓名,性別,生日,工資 行為:吃飯,跑步,教學 } 學生類設計如下: class 老師類 { 屬性:姓名,性別,生日,班級 行為:吃飯,跑步,學習 } 我們秉承著,讓最簡潔的代碼,實現最最強大的功能原則,能否讓上述案例中的重覆代碼進行優化呢?我們能 ...
一周排行
    -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中,預設只支持固定左側列,這跟大家習慣性操作列放最後不符,今天就來介紹一種簡單的方式實現固定右側列。(這裡的實現方式參考的大佬 ...