久久精品在这里_成人99免费视频_国产激情视频一区二区在线观看_国产伦精品一区二区三区免费 _亚洲午夜免费福利视频_色狠狠色狠狠综合_av在线综合网_91毛片在线观看_欧美视频一区二区在线观看_极品美女销魂一区二区三区免费_国产亚洲欧美激情_在线免费观看不卡av_日韩不卡一区二区三区_91精品国产麻豆国产自产在线_亚洲国产精品一区二区久久恐怖片_a4yy欧美一区二区三区


曙海教育集團(tuán)論壇開發(fā)語言培訓(xùn)專區(qū)Microsoft.NET Framework → 對Microsoft.NET Framework反射的反思


  共有7331人關(guān)注過本帖樹形打印

主題:對Microsoft.NET Framework反射的反思

美女呀,離線,留言給我吧!
wangxinxin
  1樓 個性首頁 | 博客 | 信息 | 搜索 | 郵箱 | 主頁 | UC


加好友 發(fā)短信
等級:青蜂俠 帖子:1393 積分:14038 威望:0 精華:0 注冊:2010-11-12 11:08:23
對Microsoft.NET Framework反射的反思  發(fā)帖心情 Post By:2010-12-14 16:28:21

清晰的組件化目標(biāo)是否因在庫間共享過多類型信息而落空?或許您需要高效的強(qiáng)類型化數(shù)據(jù)存儲,但如果每次對象模型發(fā)展后都需要更新您的數(shù)據(jù)庫架構(gòu),那會耗費(fèi)很大成本,所以您更愿意在運(yùn)行時推斷出其類型架構(gòu)嗎?您需要交付能接受任意用戶對象的組件,并以某種智能化的方式處理它們嗎?您希望庫的調(diào)方者能以編程方式向您說明它們的類型嗎?

    點(diǎn)擊下載Microsoft .NET Framework

    如果您發(fā)現(xiàn)自己在苦苦維持強(qiáng)類型化數(shù)據(jù)結(jié)構(gòu)的同時,又冀望于最大化運(yùn)行時靈活性,那么您大概會愿意考慮反射,以及它如何改善您的軟件。在本專欄中,我將探討 Microsoft .NET Framework 中的 System.Reflection 命名空間,以及它如何為您的開發(fā)體驗(yàn)提供助益。我將從一些簡單的示例開始,最后將講述如何處理現(xiàn)實(shí)世界中的序列化情形。在此過程中,我會展示反射和 CodeDom 如何配合工作,以有效處理運(yùn)行時數(shù)據(jù)。

    在深入探究 System.Reflection 之前,我想先討論一下一般的反射編程。首先,反射可定義為由一個編程系統(tǒng)提供的任何功能,此功能使程序員可以在無需提前了解其標(biāo)識或正式結(jié)構(gòu)的情況下檢查和操作代碼實(shí)體。這部分內(nèi)容很多,我將逐一展開說明。

    首先,反射提供了什么呢?您能用它做些什么呢?我傾向于將典型的以反射為中心的任務(wù)分為兩類:檢查和操作。檢查需要分析對象和類型,以收集有關(guān)其定義和行為的結(jié)構(gòu)化信息。除了一些基本規(guī)定之外,通常這是在事先不了解它們的情況下進(jìn)行的。(例如,在 .NET Framework 中,任何東西都繼承自 System.Object,并且一個對象類型的引用通常是反射的一般起點(diǎn)。)

    操作利用通過檢查收集到的信息動態(tài)地調(diào)用代碼,創(chuàng)建已發(fā)現(xiàn)類型的新實(shí)例,或者甚至可以輕松地動態(tài)重新結(jié)構(gòu)化類型和對象。需要指出的一個要點(diǎn)是,對于大多數(shù)系統(tǒng),在運(yùn)行時操作類型和對象,較之在源代碼中靜態(tài)地進(jìn)行同等操作,會導(dǎo)致性能降低。由于反射的動態(tài)特性,因此這是個必要的取舍,不過有很多技巧和最佳做法可以優(yōu)化反射的性能。

    那么,什么是反射的目標(biāo)呢?程序員實(shí)際檢查和操作什么呢?在我對反射的定義中,我用了“代碼實(shí)體”這個新術(shù)語,以強(qiáng)調(diào)一個事實(shí):從程序員的角度來說,反射技術(shù)有時會使傳統(tǒng)對象和類型之間的界限變得模糊。例如,一個典型的以反射為中心的任務(wù)可能是:

    從對象 O 的句柄開始,并使用反射獲得其相關(guān)定義(類型 T)的句柄。

    檢查類型 T,獲得它的方法 M 的句柄。

    調(diào)用另一個對象 O’(同樣是類型 T)的方法 M。

    請注意,我在從一個實(shí)例穿梭到它的底層類型,從這一類型到一個方法,之后又使用此方法的句柄在另一個實(shí)例上調(diào)用它 — 顯然這是在源代碼中使用傳統(tǒng)的 c# 編程技術(shù)無法實(shí)現(xiàn)的。在下文中探討 .NET Framework 的 System.Reflection 之后,我會再次通過一個具體的例子來解釋這一情形。

    某些編程語言本身可以通過語法提供反射,而另一些平臺和框架(如 .NET Framework)則將其作為系統(tǒng)庫。不管以何種方式提供反射,在給定情形下使用反射技術(shù)的可能性相當(dāng)復(fù)雜。編程系統(tǒng)提供反射的能力取決于諸多因素:程序員很好地利用了編程語言的功能表達(dá)了他的概念嗎?編譯器是否在輸出中嵌入足夠的結(jié)構(gòu)化信息(元數(shù)據(jù)),以方便日后的解讀?有沒有一個運(yùn)行時子系統(tǒng)或主機(jī)解釋器來消化這些元數(shù)據(jù)?平臺庫是否以對程序員有用的方式,展示此解釋結(jié)果?

    如果您頭腦中想象的是一個復(fù)雜的、面向?qū)ο箢愋偷南到y(tǒng),但在代碼中卻表現(xiàn)為簡單的、C 語言風(fēng)格的函數(shù),而且沒有正式的數(shù)據(jù)結(jié)構(gòu),那么顯然您的程序不可能動態(tài)地推斷出,某變量 v1 的指針指向某種類型 T 的對象實(shí)例。因?yàn)楫吘诡愋?T 是您頭腦中的概念,它從未在您的編程語句中明確地出現(xiàn)。但如果您使用一種更為靈活的面向?qū)ο笳Z言(如 C#)來表達(dá)程序的抽象結(jié)構(gòu),并直接引入類型 T 的概念,那么編譯器就會把您的想法轉(zhuǎn)換成某種日后可以通過合適的邏輯來理解的形式,就象公共語言運(yùn)行時 (CLR) 或某種動態(tài)語言解釋器所提供的一樣。

    反射完全是動態(tài)、運(yùn)行時的技術(shù)嗎?簡單的說,不是這樣。整個開發(fā)和執(zhí)行周期中,很多時候反射對開發(fā)人員都可用且有用。一些編程語言通過獨(dú)立編譯器實(shí)現(xiàn),這些編譯器將高級代碼直接轉(zhuǎn)換成機(jī)器能夠識別的指令。輸出文件只包括編譯過的輸入,并且運(yùn)行時沒有用于接受不透明對象并動態(tài)分析其定義的支持邏輯。這正是許多傳統(tǒng) C 編譯器的情形。因?yàn)樵谀繕?biāo)可執(zhí)行文件中幾乎沒有支持邏輯,因此您無法完成太多動態(tài)反射,然而編譯器會不時提供靜態(tài)反射 — 例如,普遍運(yùn)用的 typeof 運(yùn)算符允許程序員在編譯時檢查類型標(biāo)識。

    另一種完全不同的情況是,解釋性編程語言總是通過主進(jìn)程獲得執(zhí)行(本語言通常屬于此類)。由于程序的完整定義是可用的(作為輸入源代碼),并跟完整的語言實(shí)現(xiàn)結(jié)合在一起(作為解釋器本身),因此所有支持自我分析所需的技術(shù)都到位了。這種動態(tài)語言頻繁地提供全面反射功能,以及一組用于動態(tài)分析和操作程序的豐富工具。

    .NET Framework CLR 和它的承載語言如 C# 屬于中間形態(tài)。編譯器用來把源代碼轉(zhuǎn)換成 IL 和元數(shù)據(jù),后者與源代碼相比雖屬于較低級別或者較低“邏輯性”,但仍然保留了很多抽象結(jié)構(gòu)和類型信息。一旦 CLR 啟動和承載了此程序,基類庫 (BCL) 的 System.Reflection 庫便可以使用此信息,并返回關(guān)于對象類型、類型成員、成員簽名等的信息。此外,它也可以支持調(diào)用,包括后期綁定調(diào)用。

    .NET 中的反射

    要在用 .NET Framework 編程時利用反射,您可以使用 System.Reflection 命名空間。此命名空間提供封裝了很多運(yùn)行時概念的類,例如程序集、模塊、類型、方法、構(gòu)造函數(shù)、字段和屬性。圖 1 中的表顯示,System.Reflection 中的類如何與概念上運(yùn)行時的對應(yīng)項(xiàng)對應(yīng)起來。

    盡管很重要,不過 System.Reflection.Assembly 和 System.Reflection.Module 主要用于定位新代碼并將其加載到運(yùn)行時。本專欄中,我暫不討論這些部分,并且假定所有相關(guān)代碼都已經(jīng)加載。

    要檢查和操作已加載代碼,典型模式主要是 System.Type。通常,您從獲得一個所關(guān)注運(yùn)行時類別的 System.Type 實(shí)例開始(通過 Object.GetType)。接著您可以使用 System.Type 的各種方法,在 System.Reflection 中探索類型的定義并獲得其它類的實(shí)例。例如,如果您對某特定方法感興趣,并希望獲得此方法的一個 System.Reflection.MethodInfo 實(shí)例(可能通過 Type.GetMethod)。同樣,如果您對某字段感興趣,并希望獲得此字段的一個 System.Reflection.FieldInfo 實(shí)例(可能通過 Type.GetField)。

    一旦獲得所有必要的反射實(shí)例對象,即可根據(jù)需要遵循檢查或操作的步驟繼續(xù)。檢查時,您在反射類中使用各種描述性屬性,獲得您需要的信息(這是通用類型嗎?這是實(shí)例方法嗎?)。操作時,您可以動態(tài)地調(diào)用并執(zhí)行方法,通過調(diào)用構(gòu)造函數(shù)創(chuàng)建新對象,等等。

    檢查類型和成員

    讓我們跳轉(zhuǎn)到一些代碼中,探索如何運(yùn)用基本反射進(jìn)行檢查。我將集中討論類型分析。從一個對象開始,我將檢索它的類型,而后考察幾個有意思的成員。

    首先需要注意的是,在類定義中,乍看起來說明方法的篇幅比我預(yù)期的要多很多。這些額外的方法是從哪里來的呢?任何精通 .NET Framework 對象層次結(jié)構(gòu)的人,都會識別從通用基類 Object 自身繼承的這些方法。(事實(shí)上,我首先使用了 Object.GetType 檢索其類型。)此外,您可以看到屬性的 getter 函數(shù)。現(xiàn)在,如果您只需要 MyClass 自身顯式定義的函數(shù),該怎么辦呢?換句話說,您如何隱藏繼承的函數(shù)?或者您可能只需要顯式定義的實(shí)例函數(shù)?

    隨便在線看看 MSDN,就會發(fā)現(xiàn)大家都愿意使用 GetMethods 第二個重載方法,它接受 BindingFlags 參數(shù)。通過結(jié)合來自 BindingFlags 枚舉中不同的值,您可以讓函數(shù)僅返回所需的方法子集。替換 GetMethods 調(diào)用,代之以:

    GetMethods(BindingFlags.Instance | BindingFlags.DeclaredOnly |BindingFlags.Public)

    結(jié)果是,您得到以下輸出(注意這里不存在靜態(tài)幫助器函數(shù)和繼承自 System.Object 的函數(shù))。


以下為引用的內(nèi)容:

   Reflection Demo Example 1

  Type Name: MyClass

  Method Name: MyMethod1

  Method Name: MyMethod2

  Method Name: get_MyProperty

  Property Name: MyProperty

    如果您事先知道類型名稱(完全限定)和成員,又該如何?您如何完成從枚舉類型向檢索類型的轉(zhuǎn)換?有了前兩個示例中的代碼,您已經(jīng)有了能夠?qū)崿F(xiàn)基元類瀏覽器的基本組件。通過名稱您可以找到一個運(yùn)行時實(shí)體,然后枚舉其各種相關(guān)屬性。

[#page_動態(tài)調(diào)用代碼#0#0#0#0#]

    動態(tài)調(diào)用代碼

    迄今為止,我已經(jīng)獲得運(yùn)行時對象的句柄(如類型和方法),僅作描述用,例如輸出它們的名稱。但是如何做得更多呢?如何實(shí)際調(diào)用某個方法呢?

    此例的幾個要點(diǎn)是:首先,從一個 MyClass, mc1 實(shí)例檢索一個 System.Type 實(shí)例,然后,從該類型檢索一個 MethodInfo 實(shí)例。最后,當(dāng)調(diào)用 MethodInfo 時,通過把它作為調(diào)用的第一個參數(shù)來傳遞,將其綁定到另一個 MyClass (mc2) 實(shí)例中。

    前面講過,對于您預(yù)期在源代碼中見到的類型和對象使用之間的區(qū)別,這個示例使這種區(qū)別變得模糊。邏輯上,您檢索了一個方法的句柄,然后調(diào)用該方法,就象它屬于一個不同的對象一樣。對于熟悉函數(shù)式編程語言的程序員來說,這可能輕而易舉;但對于只熟悉 C# 的程序員來說,要分離對象實(shí)現(xiàn)和對象實(shí)例化,可能就不是那么直觀了。

    組合在一起

    至此我已經(jīng)探討過檢查和調(diào)用的基本原理,接下來我會用具體的例子把它們組合在一起。設(shè)想您希望交付一個庫,帶有必須處理對象的靜態(tài)幫助器函數(shù)。但在設(shè)計(jì)的時候,您對這些對象的類型沒有任何概念!這要看函數(shù)調(diào)用方的指示,看他希望如何從這些對象中提取有意義的信息。函數(shù)將接受一個對象集合,和一個方法的字符串描述符。然后它將遍歷該集合,調(diào)用每個對象的方法,用一些函數(shù)聚合返回值。

    就此例而言,我要聲明一些約束條件。首先,字符串參數(shù)描述的方法(必須由每個對象的底層類型實(shí)現(xiàn))不會接受任何參數(shù),并將返回一個整數(shù)。代碼將遍歷對象集合,調(diào)用指定的方法,逐步計(jì)算出所有值的平均值。最后,因?yàn)檫@不是生產(chǎn)代碼,在求和的時候我不用擔(dān)心參數(shù)驗(yàn)證或整數(shù)溢出。

    在瀏覽示例代碼時,可以看到主函數(shù)與靜態(tài)幫助器 ComputeAverage 之間的協(xié)議除了對象自身的通用基類之外,并不依賴任何類型信息。換句話說,您可以徹底改變正在傳送的對象的類型和結(jié)構(gòu),但只要總是能使用字符串描述一個方法,且該方法返回整數(shù),ComputeAverage 就可以正常工作!

    需要注意的一個關(guān)鍵問題跟隱藏在最后這個例子中的 MethodInfo(一般反射)有關(guān)。注意,在 ComputeAverage 的 foreach 循環(huán)中,代碼只從集合中的第一個對象中抓取一個 MethodInfo,然后綁定用于所有后續(xù)對象的調(diào)用。正如編碼所示,它運(yùn)行良好 — 這是 MethodInfo 緩存的一個簡單例子。但此處有一個根本性的局限。MethodInfo 實(shí)例僅能由其檢索對象同等層級類型的實(shí)例調(diào)用。因?yàn)閭魅肓?IntReturner 和 SonOfIntReturner(繼承自 IntReturner)的實(shí)例,才能這樣運(yùn)行。

    在示例代碼中,已經(jīng)包含了名為 EnemyOfIntReturner 的類,它實(shí)現(xiàn)了與其他兩個類相同的基本協(xié)議,但并沒有共享任何常見共享類型。換句話說,該接口邏輯上等同,但在類型層級上沒有重疊。要探討 MethodInfo 在該情形下的使用,請嘗試向集合添加其他對象,通過“new EnemyOfIntReturner(10)”得到一個實(shí)例,再次運(yùn)行示例。您會遇到一個異常,指出 MethodInfo 不能用于調(diào)用指定的對象,因?yàn)樗瞳@得 MethodInfo 時的原始類型完全無關(guān)(即使方法名稱和基本協(xié)議是等同的)。要使您的代碼達(dá)到生產(chǎn)水準(zhǔn),您需要做好遇到這一情形的準(zhǔn)備


支持(0中立(0反對(0單帖管理 | 引用 | 回復(fù) 回到頂部

返回版面帖子列表

對Microsoft.NET Framework反射的反思








簽名
久久精品在这里_成人99免费视频_国产激情视频一区二区在线观看_国产伦精品一区二区三区免费 _亚洲午夜免费福利视频_色狠狠色狠狠综合_av在线综合网_91毛片在线观看_欧美视频一区二区在线观看_极品美女销魂一区二区三区免费_国产亚洲欧美激情_在线免费观看不卡av_日韩不卡一区二区三区_91精品国产麻豆国产自产在线_亚洲国产精品一区二区久久恐怖片_a4yy欧美一区二区三区
性做久久久久久免费观看| 成人av在线资源网| 日韩一区二区在线观看视频| 色综合中文字幕国产| 99久久99| 99在线视频播放| 高清国产一区| 久久综合九色欧美狠狠| 日韩免费一区二区三区| 中文字幕日韩精品久久| 在线观看视频欧美| 91精品国产综合久久香蕉麻豆 | 亚洲国产一区在线观看| 亚洲国产视频直播| 美日韩一级片在线观看| 国产一区二区久久| 91亚洲精华国产精华精华液| 久久精品aaaaaa毛片| 亚洲一区二区在| 欧美日韩一区二区三区四区 | 悠悠色在线精品| 视频一区二区三区中文字幕| 麻豆91在线播放| youjizz国产精品| 精品久久精品久久| 亚洲欧洲久久| 欧美一级视频精品观看| 欧美激情在线免费观看| 伊人婷婷欧美激情| 精品亚洲aⅴ乱码一区二区三区| 国产91丝袜在线播放| 国产美女99p| 亚洲午夜高清视频| 欧美一级淫片007| 国产精品激情偷乱一区二区∴| 一区二区三区欧美| 国产精品自拍一区| 韩国成人av| 欧美日韩在线不卡| 国产精品热久久久久夜色精品三区 | 麻豆视频一区二区| 91久久国产自产拍夜夜嗨| 日韩欧美电影一区二区| 91精品国产aⅴ一区二区| 中文字幕中文字幕中文字幕亚洲无线| 石原莉奈在线亚洲二区| 99国产精品久久久久久久久久久| 日本精品一区二区| 欧美成人性战久久| 午夜精品aaa| 91免费在线看| 在线观看亚洲专区| 亚洲日本青草视频在线怡红院| 国模娜娜一区二区三区| 女同一区二区| www国产精品av| 欧美a一区二区| 久久精品欧美| 精品91自产拍在线观看一区| 爽好久久久欧美精品| 99在线观看视频网站| 欧美亚男人的天堂| 亚洲三级在线播放| 成人黄色大片在线观看| 色综合中文字幕| 亚洲欧美国产毛片在线| av在线综合网| 欧美日韩国产精选| 五月综合激情日本mⅴ| 国产日韩久久| 26uuu精品一区二区| 久久99国产精品久久99| 色视频一区二区三区| 国产精品乱码人人做人人爱| 国产一区二区福利| 尤物国产精品| 一区二区三区在线视频观看58| 91麻豆精东视频| 51午夜精品国产| 免费观看久久久4p| 五月天久久狠狠| 亚洲免费观看在线观看| 国产精品区一区| 久久综合99re88久久爱| 国产河南妇女毛片精品久久久| 色婷婷国产精品久久包臀| 亚洲乱码国产乱码精品精98午夜| 91老师片黄在线观看| 日韩丝袜情趣美女图片| 国产原创一区二区三区| 色屁屁一区二区| 日韩经典中文字幕一区| 亚洲国产精品久久久久婷婷老年| 中文字幕在线观看不卡| 成人国产1314www色视频| 久久免费视频色| 3d动漫精品啪啪一区二区三区免费| 欧美v日韩v国产v| 成人免费黄色大片| 精品国产免费久久 | 国产精品免费一区二区三区观看| 久久影院电视剧免费观看| zzijzzij亚洲日本少妇熟睡| 欧美第一区第二区| 成人开心网精品视频| 欧美成人aa大片| 99视频热这里只有精品免费| 精品国产伦一区二区三区免费| 成人手机在线视频| 精品国产乱码久久久久久久| 成人国产精品免费网站| 久久众筹精品私拍模特| 成人9ⅰ免费影视网站| 中文字幕一区二区三中文字幕| 精品日本一区二区| 一区二区国产盗摄色噜噜| 午夜一区二区三区| 蜜桃视频在线一区| 欧美丰满美乳xxx高潮www| 国产 欧美在线| 国产婷婷色一区二区三区四区| 国产精品免费一区二区三区观看| 亚洲男人的天堂av| 一本大道久久a久久综合婷婷| 激情综合五月婷婷| 精品国产凹凸成av人导航| 岛国视频一区免费观看| 一区二区三区四区乱视频| 91精品福利在线| 成人一级黄色片| 国产精品色在线| 亚洲一区二区不卡视频| 国产一区二区三区电影在线观看 | 欧美精品视频www在线观看| 粉嫩嫩av羞羞动漫久久久 | 久久99精品久久久久久久久久久久| 欧美熟乱第一页| 99re这里只有精品6| 亚洲另类在线视频| 欧美做爰猛烈大尺度电影无法无天| 国产精品羞羞答答xxdd| 国产精品欧美一区喷水| 亚洲激情电影在线| 国产a区久久久| 亚洲三级免费观看| 欧美日韩久久一区| 国产精品三区www17con| 亚洲福中文字幕伊人影院| 3atv一区二区三区| 九九九九久久久久| 精品一二三四区| 最新久久zyz资源站| 欧美视频一区在线观看| 国产精品免费区二区三区观看 | 成人高清视频在线观看| 亚洲区小说区图片区qvod| 欧美网站大全在线观看| av一区二区在线看| 蜜桃av一区二区三区电影| 久久蜜臀中文字幕| 色噜噜久久综合| 国产一区二区高清视频| 久久精品国产亚洲高清剧情介绍 | 精品一区二区三区av| 国产欧美日韩精品一区| 色狠狠色狠狠综合| 国产精品国产三级国产专区53| 热久久国产精品| 综合久久久久久| 欧美一区二区三区在线观看视频| 久久人人九九| 成人av动漫在线| 日韩经典一区二区| 中文字幕一区二区在线播放| 欧美高清视频www夜色资源网| 欧美日韩综合网| 91色porny在线视频| 国产一区在线视频| 亚洲mv在线观看| 国产精品免费久久| 日韩欧美一二区| 91九色最新地址| 视频一区二区精品| 国产嫩草一区二区三区在线观看| 国产精品911| 日本亚洲三级在线| 亚洲欧美成aⅴ人在线观看 | 亚洲视频一二区| 久久亚洲综合色| 3d成人h动漫网站入口| 中文字幕久久综合| 女女同性女同一区二区三区91| 91在线看国产| 国产夫妻精品视频| 久久国产尿小便嘘嘘尿| 午夜伊人狠狠久久| 国产精品成人免费在线| 欧美成人性福生活免费看| 欧美日韩一区二区三区四区五区| 永久久久久久|