VB.NET實(shí)現(xiàn)接口相關(guān)操作方法詳解
開(kāi)發(fā)人員在應(yīng)用VB.NET進(jìn)行實(shí)際開(kāi)發(fā)的時(shí)候,發(fā)現(xiàn)其中有不少跟其他語(yǔ)言不一樣的地方,比如在繼承方面。只支持單繼承的VB.NET為了解決多繼承的問(wèn)題,引入了接口的概念。我們可以這樣來(lái)為接口下一個(gè)定義:VB.NET實(shí)現(xiàn)接口就是指只包含虛成員的虛類(lèi)。 #t#
(1) 虛類(lèi),表明了接口是不能夠被直接實(shí)例化的。也就是說(shuō),接口只是一個(gè)抽象概念。比如我們說(shuō)車(chē)可以跑,人可以跑,馬可以跑。我們可以看到實(shí)例化的車(chē)、人、馬。但是我們可以定義一個(gè)“會(huì)跑的物質(zhì)”。他可以是車(chē),也可以使人、馬,但是我們卻不能說(shuō)“這個(gè)東西就是一個(gè)會(huì)跑得物質(zhì),但是它并不是車(chē)、人或者馬。”
(2) 只包含虛成員,表明了接口只是說(shuō)明了它具有什么樣的功能,可以提供什么樣的信息。但是這些功能和信息究竟是什么,如何提供我們無(wú)法得知。就像是“會(huì)跑的物質(zhì)”,我們知道它可以跑,但是具體他怎么跑我們就不知道了。
之所以說(shuō)VB.NET實(shí)現(xiàn)接口可以部分替代多繼承,就是因?yàn)閂B.NET只允許一個(gè)類(lèi)繼承自另一個(gè),且只能是這個(gè)類(lèi);但是一個(gè)類(lèi)可以實(shí)現(xiàn)一個(gè)或多個(gè)接口。由于接口不實(shí)現(xiàn)成員,只聲名成員,所以也就不存在多繼承的路徑問(wèn)題了。
現(xiàn)在我們假定您已經(jīng)知道了接口的聲名以及相關(guān)的一些基本知識(shí),我們來(lái)看看什么時(shí)候我們需要使用接口。
當(dāng)我們面臨一個(gè)問(wèn)題,就是我們有一個(gè)功能,它需要操作不同的類(lèi)的實(shí)例去完成一個(gè)目的相同的方法的時(shí)候,我們就可以把這些目的相同的方法作為接口來(lái)實(shí)現(xiàn)。現(xiàn)在我們看看我們面臨的問(wèn)題。目前我們手頭有一些類(lèi),它們之間沒(méi)有繼承關(guān)系,但是這些類(lèi)都可以被顯示成字符串。
- '圖書(shū)類(lèi)。可以顯示的是書(shū)名。
- Public Class Book
- Inherits Media
- Private m_Name As String
- Public Function Display() As String
- Return m_Name
- End Function
- End Class
- 'LCD顯示器類(lèi),可以顯示的是顯示器屏幕上面的內(nèi)容。
- Public Class LCD
- Inherits ComputerService
- Private m_DisplayComment As String
- Public Function Display() As String
- Return m_DisplayComment
- End Function
- End Class
- '用戶(hù)類(lèi),顯示的是全名(姓 + 名)。
- Public Class User
- Inherits Person
- Private m_FirstName, m_LastName As String
- Public Function Display () As String
- Return m_FirstName & "." & m_LastName
- End Function
- End Class
現(xiàn)在我們希望我們的程序(函數(shù))能夠把這些顯示內(nèi)容通過(guò)Console輸出到控制臺(tái)上面。由于它們不是同一個(gè)類(lèi)繼承的,所以我么現(xiàn)在VB.NET實(shí)現(xiàn)接口中有兩種選擇。
(1) 為每一個(gè)類(lèi)做一個(gè)函數(shù),分別對(duì)應(yīng)著一個(gè)類(lèi)的顯示函數(shù)。
(2) 使用一個(gè)函數(shù),用Object代替這些類(lèi),使用晚期綁定實(shí)現(xiàn)。
現(xiàn)在看看這兩種做法的問(wèn)題。
(1) 代碼復(fù)雜,而且如果新加入了別的類(lèi),我們不得不在做一個(gè)函數(shù)。
(2) 不安全。如果開(kāi)發(fā)者傳遞了一個(gè)沒(méi)有相應(yīng)方法的實(shí)例進(jìn)取就會(huì)引發(fā)異常。
現(xiàn)在我們使用接口看看。接口是不依照類(lèi)的繼承關(guān)系存在的,所以我們需要首先定義一個(gè)接口。它包含了一個(gè)Display方法。這說(shuō)明了符合這個(gè)接口的所有實(shí)例必然有這樣的一個(gè)方法,名字叫做Display,沒(méi)有參數(shù),返回字符串。
- Public Interface IDisplayer
- Function Display() As String
- End Interface
這個(gè)Display方法只是一個(gè)虛函數(shù),沒(méi)有內(nèi)容,因?yàn)槲覀儾⒉恢浪麄儜?yīng)該怎么被Display。但是我們能夠保證,他可以被Display。這樣就足夠了。現(xiàn)在我們使用這個(gè)接口來(lái)封裝我們的三個(gè)類(lèi)。讓他們實(shí)現(xiàn)這個(gè)接口,實(shí)現(xiàn)的同時(shí)我們也必須實(shí)現(xiàn)接口里面的所有虛程序。這相當(dāng)于告訴編譯器,我的類(lèi)符合接口規(guī)定的功能,我能Display,我來(lái)告訴你怎樣Display。
- '圖書(shū)類(lèi)。可以顯示的是書(shū)名。
- Public Class Book
- Inherits Media
- Implements IDisplayer
- Private m_Name As String
- Public Function Display() As String
Implements IDisplayer.Display- Return m_Name
- End Function
- End Class
- ' LCD顯示器類(lèi),可以顯示的是顯示器屏幕上面的內(nèi)容。
- Public Class LCD
- Inherits ComputerService
- Implements IDisplayer
- Private m_DisplayComment As String
- Public Function Display () As String
Implements IDisplayer.Display- Return m_DisplayComment
- End Function
- End Class
- '用戶(hù)類(lèi),顯示的是全名(姓 + 名)。
- Public Class User
- Inherits Person
- Implements IDisplayer
- Private m_FirstName, m_LastName As String
- Public Function Display() As String
Implements IDisplayer.Display- Return m_FirstName & "." & m_LastName
- End Function
- End Class
現(xiàn)在我們著手做我們的顯示函數(shù)。
- Public Sub Display
(ByVal idr As IDisplayer)- MsgBox(idr.Display)
- End Sub
我們?cè)赩B.NET實(shí)現(xiàn)接口中使用了參數(shù)idr,這個(gè)參數(shù)的類(lèi)型是一個(gè)接口IDisplayer。我們使用接口可以像使用類(lèi)一樣。實(shí)際上我們傳遞進(jìn)來(lái)的是實(shí)現(xiàn)了這個(gè)接口的某個(gè)類(lèi)的實(shí)例,但是這并不是我們關(guān)心的。我們只要知道,這個(gè)類(lèi)可以Display就足夠了。所以我么只需要直接調(diào)用接口函數(shù)Display,就可以調(diào)用到這個(gè)接口實(shí)例里面的Display函數(shù)。他肯定存在,因?yàn)樗麑?shí)現(xiàn)了接口。如果不存在,編譯器就會(huì)報(bào)錯(cuò)的。這樣我們就可以在不知道實(shí)例類(lèi)型的情況下使用方法了,而且它很安全。
如果我們需要加入一個(gè)新的類(lèi),比如是Company類(lèi),我們只要讓他也實(shí)現(xiàn)了這個(gè)接口,就可以直接適用這個(gè)函數(shù)了。
接口也允許繼承,而且允許多繼承,但是接口只能從接口繼承。比如我們的IDisplayer接口繼承了兩個(gè).NET的接口。
- Public Interface IDisplayer
- Inherits ICloneable, IComparer
- Function Display() As String
- End Interface
一個(gè)是ICloneable,他表示我們的接口支持復(fù)制(克隆);另一個(gè)是IComparer,他表示我們的接口支持比較。
現(xiàn)在我們這三個(gè)類(lèi)就出現(xiàn)了編譯錯(cuò)誤,因?yàn)槲覀儸F(xiàn)在只實(shí)現(xiàn)了IDisplayer的虛函數(shù)Display,基接口的虛函數(shù)我們還沒(méi)有實(shí)現(xiàn)。所以我們的還必須實(shí)現(xiàn)基接口的虛成員。我們以Book為例,需要稍加改動(dòng)。
- '圖書(shū)類(lèi)。可以顯示的是書(shū)名。
- Public Class Book
- Inherits Media
- Implements IDisplayer
- Private m_Name As String
- Public Sub New(ByVal Name As String)
- m_Name = Name
- End Sub
- Public Function Display1() As String
Implements IDisplayer.Display- Return m_Name
- End Function
- Public Function Compare(ByVal x As Object,
ByVal y As Object) As Integer Implements
System.Collections.IComparer.Compare- Dim bx, by As Book
- If TypeOf x Is Book AndAlso TypeOf y Is Book Then
- bx = CType(x, Book)
- by = CType(y, Book)
- Return String.Compare(bx.m_Name, by.m_Name)
- End If
- End Function
- Public Function Clone() As Object
Implements System.ICloneable.Clone- Return New Book(m_Name)
- End Function
- End Class
- 圖書(shū)類(lèi)實(shí)際上包含了三個(gè)接口:IDisplayer、
ICloneable和IComparer。但是我們使用的時(shí)候,
ICloneable和IComparer接口不會(huì)出現(xiàn),它的函數(shù)會(huì)
被當(dāng)作IDisplayer來(lái)實(shí)現(xiàn)。- Public Sub Display(ByVal idr As IDisplayer)
- MsgBox(idr.Display)
- Dim o As Object = idr.Clone
- End Sub
當(dāng)我們發(fā)現(xiàn)一些毫不相干的類(lèi),卻有一個(gè)共同的操作,他的參數(shù)和返回值一致,而我們恰恰要在某一個(gè)(或幾個(gè))地方頻繁的使用的時(shí)候,我們不妨將這些相同的部分用接口實(shí)現(xiàn)。但是前提條件是這些操作來(lái)設(shè)計(jì)邏輯來(lái)講卻是屬于相同的操作。不要為了VB.NET實(shí)現(xiàn)接口而使用它。