C#静态类 转载:(原文:http://www.cnblogs.com/chenlulouis/ )
靜態(tài)類是不能實(shí)例化的,我們直接使用它的屬性與方法,靜態(tài)類最大的特點(diǎn)就是共享。
探究
public static class StaticTestClass{
??? public static int n = 0;
??? public static void Add()
??? {
??????? n++;
??? }
}
- 網(wǎng)頁 P1.aspx 調(diào)用 StaticTestClass.Add(),并在頁面上輸出 n。
- 網(wǎng)頁 P2.aspx 調(diào)用 StaticTestClass.Add(),并在頁面上輸出 n。
?
- 訪問者 V1 從客戶端 C1 訪問 P1.aspx,此時輸出為 1。
- 訪問者 V2 從客戶端 C2 訪問 P2.aspx,此時輸出為 2。
- 訪問者 V1 關(guān)閉瀏覽器,重新打開訪問 P1.aspx,此時輸出為 3。
只要 StaticTestClass 沒有被重新編譯,即使 P1.aspx、P2.aspx 被重新編譯,每當(dāng)調(diào)用 StaticTestClass.Add(),n?都會在前一個次的基礎(chǔ)上加 1。
原則
- 靜態(tài)類中的所有成員必須是靜態(tài)的。
靜態(tài)構(gòu)造函數(shù)
- 靜態(tài)類可以有靜態(tài)構(gòu)造函數(shù),靜態(tài)構(gòu)造函數(shù)不可繼承。
- 靜態(tài)構(gòu)造函數(shù)可以用于靜態(tài)類,也可用于非靜態(tài)類。
- 靜態(tài)構(gòu)造函數(shù)無訪問修飾符、無參數(shù),只有一個 static 標(biāo)志。
- 靜態(tài)構(gòu)造函數(shù)不可被直接調(diào)用,當(dāng)創(chuàng)建類實(shí)例或引用任何靜態(tài)成員之前,靜態(tài)構(gòu)造函數(shù)被自動執(zhí)行,并且只執(zhí)行一次。
?
?
?
- 類與結(jié)構(gòu)的實(shí)例比較
- 類與結(jié)構(gòu)的差別
- 如何選擇結(jié)構(gòu)還是類
類與結(jié)構(gòu)的示例比較
結(jié)構(gòu)示例
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
??? }
}
調(diào)用過程
public class Test{
??? public static ovid Main
??? {
??????? Person Myperson=new Person????? //聲明結(jié)構(gòu)
??????? TestTime Mytime=New TestTime??? //聲明類
??? }
}
從上面的例子中我們可以看到,類的聲明和結(jié)構(gòu)的聲明非常類似,只是限定符后面是 struct 還是 class 的區(qū)別,而且使用時,定義新的結(jié)構(gòu)和定義新的類的方法也非常類似。那么類和結(jié)構(gòu)的具體區(qū)別是什么呢?
類與結(jié)構(gòu)的差別
值類型與引用類型
結(jié)構(gòu)
結(jié)構(gòu)是值類型,值類型在堆棧上分配地址,所有的基類型都是結(jié)構(gòu)類型,例如:int 對應(yīng)System.int32 結(jié)構(gòu),string 對應(yīng) system.string 結(jié)構(gòu) ,通過使用結(jié)構(gòu)可以創(chuàng)建更多的值類型。
類
類是引用類型,引用類型在堆上分配地址。
堆棧的執(zhí)行效率要比堆的執(zhí)行效率高,可是堆棧的資源有限,不適合處理大的邏輯復(fù)雜的對象。所以結(jié)構(gòu)處理作為基類型對待的小對象,而類處理某個商業(yè)邏輯。
因?yàn)榻Y(jié)構(gòu)是值類型所以結(jié)構(gòu)之間的賦值可以創(chuàng)建新的結(jié)構(gòu),而類是引用類型,類之間的賦值只是復(fù)制引用。
說明:
- 雖然結(jié)構(gòu)與類的類型不一樣,可是他們的基類型都是對象(object),C# 中所有類型的基類型都是 Object。
- 雖然結(jié)構(gòu)的初始化也使用了 new 操作符可是結(jié)構(gòu)對象依然分配在堆棧上而不是堆上,如果不使用“新建”(new),那么在初始化所有字段之前,字段將保持未賦值狀態(tài),且對象不可用。
繼承性
結(jié)構(gòu)
不能從另外一個結(jié)構(gòu)或者類繼承,本身也不能被繼承,雖然結(jié)構(gòu)沒有明確的用 sealed 聲明,可是結(jié)構(gòu)是隱式的 sealed。
類
完全可擴(kuò)展的,除非顯示的聲明 sealed,否則類可以繼承其他類和接口,自身也能被繼承。
說明:
- 雖然結(jié)構(gòu)不能被繼承 可是結(jié)構(gòu)能夠繼承接口,方法和類繼承接口一樣。
例如:結(jié)構(gòu)實(shí)現(xiàn)接口
interface IImage
{
??? void Paint();
}
?
struct Picture : IImage
{
??? public void Paint()
??? {
???????? // painting code goes here
??? }
????
??? private int x, y, z;
?? // other struct members
}
內(nèi)部結(jié)構(gòu)
結(jié)構(gòu)
- 沒有默認(rèn)的構(gòu)造函數(shù),但是可以添加構(gòu)造函數(shù)
- 沒有析構(gòu)函數(shù)
- 沒有 abstract 和 sealed(因?yàn)椴荒芾^承)
- 不能有 protected 修飾符
- 可以不使用 new 初始化
- 在結(jié)構(gòu)中初始化實(shí)例字段是錯誤的
類
- 有默認(rèn)的構(gòu)造函數(shù)
- 有析構(gòu)函數(shù)
- 可以使用 abstract 和 sealed
- 有 protected 修飾符
- 必須使用 new 初始化
如何選擇結(jié)構(gòu)還是類
討論了結(jié)構(gòu)與類的相同之處和差別之后,下面討論如何選擇使用結(jié)構(gòu)還是類:
- 堆棧的空間有限,對于大量的邏輯的對象,創(chuàng)建類要比創(chuàng)建結(jié)構(gòu)好一些。
- 結(jié)構(gòu)表示如點(diǎn)、矩形和顏色這樣的輕量對象,例如,如果聲明一個含有 1000 個點(diǎn)對象的數(shù)組,則將為引用每個對象分配附加的內(nèi)存。在此情況下,結(jié)構(gòu)的成本較低。
- 在表現(xiàn)抽象和多級別的對象層次時,類是最好的選擇。
- 大多數(shù)情況下該類型只是一些數(shù)據(jù)時,結(jié)構(gòu)時最佳的選擇。
----
網(wǎng)友在 CSDN 上的回答:
結(jié)構(gòu)可以看作是輕量級的類,在性能上要好一點(diǎn)。
相同之處:
- 結(jié)構(gòu)和類對于程序來講都通過指針操作,同樣是面向?qū)ο蟮男问健?/li>
不同之處:
- 結(jié)構(gòu)體對象總是在線程堆棧上操作,而不是托管堆上。
- 不能繼承一個結(jié)構(gòu)體(所以在調(diào)用結(jié)構(gòu)體的方法時不需要查找 vtable: 虛函數(shù)繼承表)
- 我們不能聲明構(gòu)造函數(shù)為空的結(jié)構(gòu)體(不曉得為啥非得要這么設(shè)計(jì))
- 結(jié)構(gòu)體的構(gòu)造函數(shù)內(nèi)必須初始化所有變量(不曉得為啥非得要這么設(shè)計(jì))
- 結(jié)構(gòu)體的字段不能有默認(rèn)值(默認(rèn)都是二進(jìn)制意義上的零值),但是可以在構(gòu)造函數(shù)內(nèi)改變“默認(rèn)值”
.....
按照MSDN上的意思,實(shí)際上適合用 struct 的場合很小,結(jié)構(gòu)使用指南:
- 行為與基元類型一樣。
- 具有 16 字節(jié)以下的實(shí)例大小。
- 是不可改變的。
- 值語義是合意的。
?
在?C# 中結(jié)構(gòu)與類的區(qū)別一文中,已經(jīng)介紹了 struct 的相關(guān)知識,本文就結(jié)合應(yīng)用作些強(qiáng)調(diào)、補(bǔ)充、修正。
關(guān)于字段
不能在聲明字段時初始化它,除非字段被標(biāo)明為 const 或 static。
關(guān)于構(gòu)造函數(shù)
構(gòu)造函數(shù)必須有參數(shù)。
構(gòu)造函數(shù)中必須為所有的字段賦值。
說明
不允許在結(jié)構(gòu)中顯式地聲明無參數(shù)的構(gòu)造函數(shù),若要顯示地聲明則必須是有參數(shù)的。但在使用 new 實(shí)例化時可以使用無參數(shù)的構(gòu)造函數(shù),也可以使用有參數(shù)的構(gòu)造函數(shù),說明存在著一個看不到的、默認(rèn)的無參數(shù)構(gòu)造函數(shù),《C# 中結(jié)構(gòu)與類的區(qū)別》一文中說結(jié)構(gòu)“沒有默認(rèn)的構(gòu)造函數(shù)”,這種說法有誤。
要不要使用 new
如果使用結(jié)構(gòu)中的屬性、方法,則必須使用 new,否則可以不使用 new。
?
?
嵌套類型
?
在類或結(jié)構(gòu)內(nèi)部定義的類型稱為嵌套類型。例如:
class Container {class Nested{Nested() { }} }不管外部類型是類還是結(jié)構(gòu),嵌套類型均默認(rèn)為 private,但是可以設(shè)置為 public、protected internal、protected、internal 或 private。在上面的示例中,Nested 對外部類型是不可訪問的,但可以設(shè)置為 public,如下所示:
class Container {public class Nested{Nested() { }} }嵌套類型(或內(nèi)部類型)可訪問包含類型(或外部類型)。若要訪問包含類型,請將其作為構(gòu)造函數(shù)傳遞給嵌套類型。例如:
public class Container {public class Nested{private Container m_parent;public Nested(){}public Nested(Container parent){m_parent = parent;}} }嵌套類型可訪問包含類型的私有成員和受保護(hù)的成員(包括所有繼承的私有成員或受保護(hù)的成員)。
在前面的聲明中,類 Nested 的完整名稱為 Container.Nested。這是用來創(chuàng)建嵌套類的新實(shí)例的名稱,如下所示:
Container.Nested nest = new Container.Nested();
作者:ChenLuLouis?
出處:http://www.cnblogs.com/chenlulouis/?
本文版權(quán)歸作者和博客園共有,歡迎轉(zhuǎn)載,但未經(jīng)作者同意必須保留此段聲明,且在文章頁面明顯位置給出原文連接,否則保留追究法律責(zé)任的權(quán)利。?
該文章也同時發(fā)布在我的獨(dú)立博客中-chenlulouisBlog。
轉(zhuǎn)載于:https://www.cnblogs.com/joyang/p/5125167.html
總結(jié)
以上是生活随笔為你收集整理的C#静态类 转载:(原文:http://www.cnblogs.com/chenlulouis/ )的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: AppStore 上架注意事项及错误修改
- 下一篇: 【C语言笔记进阶篇】第二章:字符串函数和