作為.net程序員,我們每天都要和BCL(Base Class Linbrary)打交道。無疑,BCL做為一個年輕的框架類庫,她是成功的,但是還有一些時候我們還是得寫一些”Helper”方法來擴展類庫,由于我們不能修改類庫的源代碼,我們只有寫一個個的靜態類。雖然在使用上也算方便,但作為追求完美的程序員來說總有些不雅。
現在我就碰到這樣的事情,前兩天奉命寫一個從XML文件加載Chart圖的設置的方法,從XML加載數據綁定到對象上,這肯定是反射的用武之地了。我經常需要寫一些根據對象屬性名字來判斷這個對象是否有這個屬性或者根據屬性名獲取該屬性的值。還是按照平常一樣,我很快寫了一個PropertyHelper,里面有兩個靜態方法:HasProperty,GetValueByName。
PropertyHelper.HasProperty(point, "X"),如此的調用也還過得去,不過在C# 3.0微軟為我們提供了擴展方法?,F在我們可以直接這樣調用了point.HasProperty(“X”);看看我是如何實現這個擴展方法的?
public static class PropertyExtension{ public static object GetValueByName(this object self, string propertyName) { if (self == null) { return self ; } Type t = self.GetType(); PropertyInfo p = t.GetProperty(propertyName); return p.GetValue(self, null); }}
我給object類型添加了一個擴展方法,在.net里所有的類都繼承自object,那所有的類都默認的擁有這個方法了,真方便,呵呵。
注意到和普通的靜態方法有何差別?在這個方法的第一個參數前面多了一個this關鍵字。
擴展方法:
1、方法所在的類必須是靜態的
2、方法也必須是靜態的
3、方法的第一個參數必須是你要擴展的那個類型,比如你要給int擴展一個方法,那么第一個參數就必須是int。
4、在第一個參數前面還需要有一個this關鍵字。
按照上面的步驟寫你就得到了一個“擴展方法”,你可以像調用這個類的原生方法那樣去調用它:
好像string類型現在有了GetValueByName這個方法一樣,但實際上string并沒有這樣一個方法。那這又是為什么呢?是我們可愛的編譯器在其中做了手腳。為了避開編譯器的干擾,我們來直接欣賞MSIL代碼:
從MSIL中我們可以看出,這段代碼編譯后和調用靜態方法沒有任何的差別(從call指令來看,這是在調用一個靜態方法)。
從這里可以知道擴展方法即可以使用實例調用的方式也可以直接使用靜態類調用的方式:
下面將對擴展方法做一些細節的介紹:
Visual Studio 2008對擴展方法有智能感知的支持,如下圖
在方法的圖標上有一個與其他的都不相同,他的突變下面還帶有一個藍色的向下的箭頭,這就表明這個方法是一個擴展方法。
下面是對編寫擴展方法要注意的幾個原則(當然,仁者見仁、智者見智,這也是一家之言):
擴展方法有就近原則,也就是如果在你的程序里有兩個一模一樣的擴展方法,一個和你的使用類是處于同一命名空間里,另外一個處于別的命名空間里,這個時候會優先使用同一命名空間里的擴展方法,也就是說“血緣關系”越近,越被青睞。
很多人看到擴展方法也許眼里冒出金光,以后在設計的時候不管三七二十一,反正可以擴展。還有一些人會對類任意擴展,將以前一些作為”Helper”的方法統統使用擴展方法代替,注意的是擴展方法有“污染性”,所以我覺得在擴展的時候還是想想,是不是值得這樣擴展。
在擴展的時候也不要對比較高層的類進行擴展,像我上面對object的擴展我覺得就是不可取的,object是所有類的基類,一經擴展,所有的類都被“污染”了。
發表與2008-07-16
于2008-08-06第一次更新
新聞熱點
疑難解答