1. 程式人生 > >C# 靜態類

C# 靜態類

靜態類是不能例項化的,我們直接使用它的屬性與方法,靜態類最大的特點就是共享。

探究

public static class StaticTestClass
{
    public static int n = 0;

    public static void Add()
    {
        n++;
    }
}
  • 網頁 P1.aspx 呼叫 StaticTestClass.Add(),並在頁面上輸出 n。
  • 網頁 P2.aspx 呼叫 StaticTestClass.Add(),並在頁面上輸出 n。
  • 訪問者 V1 從客戶端 C1 訪問 P1.aspx,此時輸出為 1。
  • 訪問者 V2 從客戶端 C2 訪問 P2.aspx,此時輸出為 2。
  • 訪問者 V1 關閉瀏覽器,重新開啟訪問 P1.aspx,此時輸出為 3。

只要 StaticTestClass 沒有被重新編譯,即使 P1.aspx、P2.aspx 被重新編譯,每當呼叫 StaticTestClass.Add(),n 都會在前一個次的基礎上加 1。

原則

  • 靜態類中的所有成員必須是靜態的。

靜態建構函式

  • 靜態類可以有靜態建構函式,靜態建構函式不可繼承。
  • 靜態建構函式可以用於靜態類,也可用於非靜態類。
  • 靜態建構函式無訪問修飾符、無引數,只有一個 static 標誌。
  • 靜態建構函式不可被直接呼叫,當建立類例項或引用任何靜態成員之前,靜態建構函式被自動執行,並且只執行一次。
  • 類與結構的例項比較
  • 類與結構的差別
  • 如何選擇結構還是類

類與結構的示例比較

結構示例

public struct Person
{
    string Name;
    int  height;
    int  weight
   
    public bool overWeight()
    {
        //implement something     }
}

類示例

public class TestTime
{
    int hours;
    int minutes;
    int seconds;
   
    public void passtime()
    {
        //implementation of behavior
    }
}

呼叫過程

public class Test
{
    public static ovid Main
    {
        Person Myperson=new Person      //宣告結構
        TestTime Mytime=New TestTime    //宣告類
    }
}

從上面的例子中我們可以看到,類的宣告和結構的宣告非常類似,只是限定符後面是 struct 還是 class 的區別,而且使用時,定義新的結構和定義新的類的方法也非常類似。那麼類和結構的具體區別是什麼呢?

類與結構的差別

值型別與引用型別

結構

結構是值型別,值型別在堆疊上分配地址,所有的基型別都是結構型別,例如:int 對應System.int32 結構,string 對應 system.string 結構 ,通過使用結構可以建立更多的值型別。

類是引用型別,引用型別在堆上分配地址。

堆疊的執行效率要比堆的執行效率高,可是堆疊的資源有限,不適合處理大的邏輯複雜的物件。所以結構處理作為基型別對待的小物件,而類處理某個商業邏輯。

因為結構是值型別所以結構之間的賦值可以建立新的結構,而類是引用型別,類之間的賦值只是複製引用。

說明:

  • 雖然結構與類的型別不一樣,可是他們的基型別都是物件(object),C# 中所有型別的基型別都是 Object。
  • 雖然結構的初始化也使用了 new 操作符可是結構物件依然分配在堆疊上而不是堆上,如果不使用“新建”(new),那麼在初始化所有欄位之前,欄位將保持未賦值狀態,且物件不可用。

繼承性

結構

不能從另外一個結構或者類繼承,本身也不能被繼承,雖然結構沒有明確的用 sealed 宣告,可是結構是隱式的 sealed。

完全可擴充套件的,除非顯示的宣告 sealed,否則類可以繼承其他類和介面,自身也能被繼承。

說明:

  • 雖然結構不能被繼承 可是結構能夠繼承介面,方法和類繼承介面一樣。

    例如:結構實現介面

    interface IImage
    {
        void Paint();
    }
     
    struct Picture : IImage
    {
        public void Paint()
        {
             // painting code goes here
        }
       
        private int x, y, z;
       // other struct members
    }

內部結構

結構

  • 沒有預設的建構函式,但是可以新增建構函式
  • 沒有解構函式
  • 沒有 abstract 和 sealed(因為不能繼承)
  • 不能有 protected 修飾符
  • 可以不使用 new 初始化
  • 在結構中初始化例項欄位是錯誤的

  • 有預設的建構函式
  • 有解構函式
  • 可以使用 abstract 和 sealed
  • 有 protected 修飾符
  • 必須使用 new 初始化

如何選擇結構還是類

討論了結構與類的相同之處和差別之後,下面討論如何選擇使用結構還是類:

  • 堆疊的空間有限,對於大量的邏輯的物件,建立類要比建立結構好一些。
  • 結構表示如點、矩形和顏色這樣的輕量物件,例如,如果宣告一個含有 1000 個點物件的陣列,則將為引用每個物件分配附加的記憶體。在此情況下,結構的成本較低。
  • 在表現抽象和多級別的物件層次時,類是最好的選擇。
  • 大多數情況下該型別只是一些資料時,結構時最佳的選擇。

----

網友在 CSDN 上的回答:

結構可以看作是輕量級的類,在效能上要好一點。

相同之處:

  • 結構和類對於程式來講都通過指標操作,同樣是面向物件的形式。

不同之處:

  • 結構體物件總是線上程堆疊上操作,而不是託管堆上。
  • 不能繼承一個結構體(所以在呼叫結構體的方法時不需要查詢 vtable: 虛擬函式繼承表)
  • 我們不能宣告建構函式為空的結構體(不曉得為啥非得要這麼設計)
  • 結構體的建構函式內必須初始化所有變數(不曉得為啥非得要這麼設計)
  • 結構體的欄位不能有預設值(預設都是二進位制意義上的零值),但是可以在建構函式內改變“預設值”

.....

按照MSDN上的意思,實際上適合用 struct 的場合很小,結構使用指南:

  • 行為與基元型別一樣。
  • 具有 16 位元組以下的例項大小。
  • 是不可改變的。
  • 值語義是合意的。

C# 中結構與類的區別一文中,已經介紹了 struct 的相關知識,本文就結合應用作些強調、補充、修正。

關於欄位

不能在宣告欄位時初始化它,除非欄位被標明為 const 或 static。

關於建構函式

建構函式必須有引數。

建構函式中必須為所有的欄位賦值。

說明

不允許在結構中顯式地宣告無引數的建構函式,若要顯示地宣告則必須是有引數的。但在使用 new 例項化時可以使用無引數的建構函式,也可以使用有引數的建構函式,說明存在著一個看不到的、預設的無引數建構函式,《C# 中結構與類的區別》一文中說結構“沒有預設的建構函式”,這種說法有誤。

要不要使用 new

如果使用結構中的屬性、方法,則必須使用 new,否則可以不使用 new。

巢狀型別

在類或結構內部定義的型別稱為巢狀型別。例如:

class Container
{
    class Nested
    {
        Nested() { }
    }
}

不管外部型別是類還是結構,巢狀型別均預設為 private,但是可以設定為 public、protected internal、protected、internal 或 private。在上面的示例中,Nested 對外部型別是不可訪問的,但可以設定為 public,如下所示:

class Container
{
    public class Nested
    {
        Nested() { }
    }
}

巢狀型別(或內部型別)可訪問包含型別(或外部型別)。若要訪問包含型別,請將其作為建構函式傳遞給巢狀型別。例如:

public class Container
{
    public class Nested
    {
        private Container m_parent;

        public Nested()
        {
        }
        public Nested(Container parent)
        {
            m_parent = parent;
        }
    }
}

巢狀型別可訪問包含型別的私有成員和受保護的成員(包括所有繼承的私有成員或受保護的成員)。

在前面的宣告中,類 Nested 的完整名稱為 Container.Nested。這是用來建立巢狀類的新例項的名稱,如下所示:

Container.Nested nest = new Container.Nested();