一點通VB.NET構造器講解
面向對象編程現在被大多數人使用,這里我們一起來看看關于面向對象中構造器的理解。VB.NET構造器的范圍對類的行為有重要的含義。一個Public類中的Friend構造器使我們只可以從同一個程序集內部創建這個類,所以它同你在VB6的類中用的PublicNotCreatable設置有很多共同之處。一個私有的(private)VB.NET構造器使這個類根本不能創建,如果類只是共享方法的一個容器,這種構造器就很有用。
#T#(這樣的類的例子有System.Console和System.Environment。)更確切地說,一個代碼片段可以實例化一個帶有私有構造器的類,只要那個代碼位于類內部或嵌套的類中,因為一個嵌套的類型可以訪問包含它的類型的私有的成員。創建只包含一個共享成員的一個 VB.NET類的更簡單的方法就是定義一個Module。Module是規則的、不能創建的類,它的成員是靜態的。注意,.NET runtime對模塊并不很重視(C#中沒有Module):VB.NET對模塊的支持只可以簡化VB6代碼的移植,而且編譯器將一個Module中的所有成員都明確地轉換成靜態成員。
注意初始化字段,前面的講述可能意味著私有VB.NET構造器只有在很少的情況下才有用,但實際并不是這樣的。例如,當你的類包含許多字段的初始化設置時,定義一個空的Private Sub New過程就很方便:
- Public MinSize As Integer = 10
- Public MaxSize As Integer = 1000
- ' ...(other fields with initializers)
編譯器在每個構造器開始處都會進行隱含的賦值,保證在構造器運行時,所有的字段都包含正確的初始值。如果你有20個初始化字段和10個構造器,那么你的類就會包含多達200個隱含的賦值,這樣就會浪費內存中和磁盤上的字節。如果你定義一個虛擬的不帶參數的私有構造器,并讓所有的公有構造器調用它,那么編譯器就只添加20個隱含的語句到私有構造器中。通過Microsoft Intermediate Language Disassembler(ILDASM)運行產生的可執行的文件,你就可以看到在每種情況下編譯器創建的代碼。
當客戶端要通過一個共享的函數(作為類的工廠方法(factory method ))來創建類的實例時,就體現了私有構造器的另一個好處。一個共享的方法可以讓你在創建類的一個新實例前運行一些代碼——例如,查看一個具有相同屬性的對象是否在你內部管理的對象池中。你不能用一個規則的構造器來實現這種功能,因為只有在一個新實例已經運行時,規則的構造器的代碼才運行(見列表1)。
你在從一個類派生一個更簡單的新類時,可以看到OOP的強大。派生的類自動繼承基類的所有字段、屬性、事件和接口,所以你只需要關注你想添加到派生的類中的成員:
- Class Person
- Public FirstName As String
- Public LastName As String
- Function CompleteName() As String
- Return FirstName & " " & LastName
- End Function
- End Class
- Class Customer
- Inherits Person
- ' a new field and a new method
- Public Title As String
- Function ReverseName() As String
- Return LastName & ", " & FirstName
- End Function
- End Class
更好的是,如果你期望派生的類有不同的行為,你還可以覆蓋基類中的屬性或方法。例如,你可能想讓Customer.CompleteName方法以 “Mr. John Doe”的形式返回一個字符串。你必須做兩件事來覆蓋一個成員:將基類的成員標記為Overridable,使它成為一個虛擬的成員,用關鍵字 Overrides來標記派生的類的成員:
- ' in Person class
- Overridable Function CompleteName() As String
- ' ...(as before)
- End Function
- ' in Customer class
- Overrides Function CompleteName() _
- As String
- Return Title & " " & FirstName _
- & " "& LastName
- End Function
重用基類中的代碼
Visual Studio .NET為我們在一個派生的類中寫被覆蓋的成員的代碼提供了一個很好的捷徑:在編輯窗口上方最左邊的ComboBox中選擇類名字下的(Overrides)成員,然后在最右邊的ComboBox中選擇你想覆蓋的成員(見圖1)。在派生的類中你不需要用關鍵字Overridable,因為被覆蓋的方法本身就是可以被覆蓋的。如果你出于某種原因想停止進一步覆蓋那個方法,你必須用關鍵字NotOverridable標記它:
- ' derived classes can't override this
- NotOverridable Overrides Function _
- CompleteName()As String
- ' ...
- End Function