一點(diǎn)通VB.NET構(gòu)造器講解
面向?qū)ο缶幊态F(xiàn)在被大多數(shù)人使用,這里我們一起來看看關(guān)于面向?qū)ο笾袠?gòu)造器的理解。VB.NET構(gòu)造器的范圍對(duì)類的行為有重要的含義。一個(gè)Public類中的Friend構(gòu)造器使我們只可以從同一個(gè)程序集內(nèi)部創(chuàng)建這個(gè)類,所以它同你在VB6的類中用的PublicNotCreatable設(shè)置有很多共同之處。一個(gè)私有的(private)VB.NET構(gòu)造器使這個(gè)類根本不能創(chuàng)建,如果類只是共享方法的一個(gè)容器,這種構(gòu)造器就很有用。
#T#(這樣的類的例子有System.Console和System.Environment。)更確切地說,一個(gè)代碼片段可以實(shí)例化一個(gè)帶有私有構(gòu)造器的類,只要那個(gè)代碼位于類內(nèi)部或嵌套的類中,因?yàn)橐粋€(gè)嵌套的類型可以訪問包含它的類型的私有的成員。創(chuàng)建只包含一個(gè)共享成員的一個(gè) VB.NET類的更簡單的方法就是定義一個(gè)Module。Module是規(guī)則的、不能創(chuàng)建的類,它的成員是靜態(tài)的。注意,.NET runtime對(duì)模塊并不很重視(C#中沒有Module):VB.NET對(duì)模塊的支持只可以簡化VB6代碼的移植,而且編譯器將一個(gè)Module中的所有成員都明確地轉(zhuǎn)換成靜態(tài)成員。
注意初始化字段,前面的講述可能意味著私有VB.NET構(gòu)造器只有在很少的情況下才有用,但實(shí)際并不是這樣的。例如,當(dāng)你的類包含許多字段的初始化設(shè)置時(shí),定義一個(gè)空的Private Sub New過程就很方便:
- Public MinSize As Integer = 10
- Public MaxSize As Integer = 1000
- ' ...(other fields with initializers)
編譯器在每個(gè)構(gòu)造器開始處都會(huì)進(jìn)行隱含的賦值,保證在構(gòu)造器運(yùn)行時(shí),所有的字段都包含正確的初始值。如果你有20個(gè)初始化字段和10個(gè)構(gòu)造器,那么你的類就會(huì)包含多達(dá)200個(gè)隱含的賦值,這樣就會(huì)浪費(fèi)內(nèi)存中和磁盤上的字節(jié)。如果你定義一個(gè)虛擬的不帶參數(shù)的私有構(gòu)造器,并讓所有的公有構(gòu)造器調(diào)用它,那么編譯器就只添加20個(gè)隱含的語句到私有構(gòu)造器中。通過Microsoft Intermediate Language Disassembler(ILDASM)運(yùn)行產(chǎn)生的可執(zhí)行的文件,你就可以看到在每種情況下編譯器創(chuàng)建的代碼。
當(dāng)客戶端要通過一個(gè)共享的函數(shù)(作為類的工廠方法(factory method ))來創(chuàng)建類的實(shí)例時(shí),就體現(xiàn)了私有構(gòu)造器的另一個(gè)好處。一個(gè)共享的方法可以讓你在創(chuàng)建類的一個(gè)新實(shí)例前運(yùn)行一些代碼——例如,查看一個(gè)具有相同屬性的對(duì)象是否在你內(nèi)部管理的對(duì)象池中。你不能用一個(gè)規(guī)則的構(gòu)造器來實(shí)現(xiàn)這種功能,因?yàn)橹挥性谝粋€(gè)新實(shí)例已經(jīng)運(yùn)行時(shí),規(guī)則的構(gòu)造器的代碼才運(yùn)行(見列表1)。
你在從一個(gè)類派生一個(gè)更簡單的新類時(shí),可以看到OOP的強(qiáng)大。派生的類自動(dòng)繼承基類的所有字段、屬性、事件和接口,所以你只需要關(guān)注你想添加到派生的類中的成員:
- 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”的形式返回一個(gè)字符串。你必須做兩件事來覆蓋一個(gè)成員:將基類的成員標(biāo)記為Overridable,使它成為一個(gè)虛擬的成員,用關(guān)鍵字 Overrides來標(biāo)記派生的類的成員:
- ' 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為我們?cè)谝粋€(gè)派生的類中寫被覆蓋的成員的代碼提供了一個(gè)很好的捷徑:在編輯窗口上方最左邊的ComboBox中選擇類名字下的(Overrides)成員,然后在最右邊的ComboBox中選擇你想覆蓋的成員(見圖1)。在派生的類中你不需要用關(guān)鍵字Overridable,因?yàn)楸桓采w的方法本身就是可以被覆蓋的。如果你出于某種原因想停止進(jìn)一步覆蓋那個(gè)方法,你必須用關(guān)鍵字NotOverridable標(biāo)記它:
- ' derived classes can't override this
- NotOverridable Overrides Function _
- CompleteName()As String
- ' ...
- End Function