VB.net是怎樣做到的(一)
VB.net能夠實現很多C#不能做到的功能,如When語句、Optional參數、局部Static變量、對象實例訪問靜態方法、Handles綁定事件、On Error處理異常、Object直接后期綁定等等。VB和C#同屬.net的語言,編譯出來的是同樣的CIL,但為什么VB支持很多有趣的特性呢。我們一起來探究一下。
首先是局部Static變量。VB支持用Static關鍵字聲明局部變量,這樣在過程結束的時候可以保持變量的數值:
Public Sub Test1()
Static i As Integer
i += 1 '實現一個過程調用計數器
End Sub
我們實現了一個簡單的過程計數器。每調用一次Test,計數器的數值就增加1。其實還有很多情況我們希望保持變量的數值。而C#的static是不能用在過程內部的。因此要實現過程計數器,我們必須聲明一個類級別的變量。這樣做明顯不如VB好。因為無法防止其他過程修改計數器變量。這就和對象封裝一個道理,本來應該是一個方法的局部變量,現在我要被迫把它獨立出來,顯然是不好的設計。那么VB是怎么生成局部靜態變量的呢?將上述代碼返匯編,我們可以清楚地看到在VB生成的CIL中,i不是作為局部變量,而是作為類的Field出現的:
.field private specialname int32 $STATIC$Test1$2001$i
也就是說,i被改名作為一個類的字段,但被冠以specialname。在代碼中試圖訪問$STATIC$Test1$2001$i是不可能的,因為它不是一個有效的標識符。但是在IL中,將這個變量加一的代碼卻與一般的類字段完全一樣,是通過ldfld加載的。我覺得這個方法十分聰明,把靜態變量變成生命周期一樣的類字段,但是又由編譯器來控制訪問的權限,讓它成為一個局部變量。同時也解釋了VB為什么要用兩個不同的關鍵字來聲明靜態變量——Static和Shared
VB.net是怎樣做到的(二)
VB.net支持一項很有意思的功能——MyClass。大部分人使用MyClass可能僅限于調用本類其他構造函數時。其實MyClass可以產生一些很獨特的用法。MyClass永遠按類的成員為不可重寫的狀態進行調用,即當類的方法被重寫后,用MyClass仍能得到自身的版本。下面這個例子和VB幫助中所舉的例子非常相似
Public Class MyClassBase Protected Overridable Sub Greeting() Console.WriteLine("Hello form Base") End Sub
Public Sub UseMe() Me.Greeting() End Sub
Public Sub UseMyClass() MyClass.Greeting() End Sub End Class
Public Class MyClassSub Inherits MyClassBase
Protected Overrides Sub Greeting() Console.WriteLine("Hello form Sub") End Sub End Class
我們用一段代碼來測試:
Dim o As MyClassBase = New MyClassSub()
o.UseMe() o.UseMyClass()
結果是UseMe執行了子類的版本,而UseMyClass還是執行了基類本身的版本,盡管這是一個虛擬方法。觀其IL,可以看到其簡單的實現原理:
Me用的調用指令是callvirt
IL_0001: callvirt instance void App1.MyClassBase::Greeting()
而MyClass調用的是call
IL_0001: call instance void App1.MyClassBase::Greeting()
奇怪的是,如此簡單的一個功能,我竟然無法用C#實現,C#怎樣也不允許我按非虛函數的方式調用一個虛函數。C++可以用類名::方法名的方式訪問自身版本的函數,但C#的類名只能用來訪問靜態的成員。這真是C#一個奇怪的缺陷。
|