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

来源: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#TMS系統代碼-基礎頁面BaseCity學習 本人純新手,剛進公司跟領導報道,我說我是java全棧,他問我會不會C#,我說大學學過,他說這個TMS系統就給你來管了。外包已經把代碼給我了,這幾天先把增刪改查的代碼背一下,說不定後面就要趕鴨子上架了 Service頁面 //using => impo ...
  • 委托與事件 委托 委托的定義 委托是C#中的一種類型,用於存儲對方法的引用。它允許將方法作為參數傳遞給其他方法,實現回調、事件處理和動態調用等功能。通俗來講,就是委托包含方法的記憶體地址,方法匹配與委托相同的簽名,因此通過使用正確的參數類型來調用方法。 委托的特性 引用方法:委托允許存儲對方法的引用, ...
  • 前言 這幾天閑來沒事看看ABP vNext的文檔和源碼,關於關於依賴註入(屬性註入)這塊兒產生了興趣。 我們都知道。Volo.ABP 依賴註入容器使用了第三方組件Autofac實現的。有三種註入方式,構造函數註入和方法註入和屬性註入。 ABP的屬性註入原則參考如下: 這時候我就開始疑惑了,因為我知道 ...
  • C#TMS系統代碼-業務頁面ShippingNotice學習 學一個業務頁面,ok,領導開完會就被裁掉了,很突然啊,他收拾東西的時候我還以為他要旅游提前請假了,還在尋思為什麼回家連自己買的幾箱飲料都要叫跑腿帶走,怕被偷嗎?還好我在他開會之前拿了兩瓶芬達 感覺感覺前面的BaseCity差不太多,這邊的 ...
  • 概述:在C#中,通過`Expression`類、`AndAlso`和`OrElse`方法可組合兩個`Expression<Func<T, bool>>`,實現多條件動態查詢。通過創建表達式樹,可輕鬆構建複雜的查詢條件。 在C#中,可以使用AndAlso和OrElse方法組合兩個Expression< ...
  • 閑來無聊在我的Biwen.QuickApi中實現一下極簡的事件匯流排,其實代碼還是蠻簡單的,對於初學者可能有些幫助 就貼出來,有什麼不足的地方也歡迎板磚交流~ 首先定義一個事件約定的空介面 public interface IEvent{} 然後定義事件訂閱者介面 public interface I ...
  • 1. 案例 成某三甲醫預約系統, 該項目在2024年初進行上線測試,在正常運行了兩天後,業務系統報錯:The connection pool has been exhausted, either raise MaxPoolSize (currently 800) or Timeout (curren ...
  • 背景 我們有些工具在 Web 版中已經有了很好的實踐,而在 WPF 中重新開發也是一種費時費力的操作,那麼直接集成則是最省事省力的方法了。 思路解釋 為什麼要使用 WPF?莫問為什麼,老 C# 開發的堅持,另外因為 Windows 上已經裝了 Webview2/edge 整體打包比 electron ...
  • EDP是一套集組織架構,許可權框架【功能許可權,操作許可權,數據訪問許可權,WebApi許可權】,自動化日誌,動態Interface,WebApi管理等基礎功能於一體的,基於.net的企業應用開發框架。通過友好的編碼方式實現數據行、列許可權的管控。 ...
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...