詳解.NET中的代碼動態(tài)編譯_.Net教程

      編輯Tag賺U幣
      教程Tag:暫無Tag,歡迎添加,賺取U幣!

      推薦:.NET編程之關(guān)于VS 2008和.NET 3.5 Beta2新特性介紹
      非常重要的注意事項: 請閱讀本博客貼子下面的“安裝注意事項”,內(nèi)含安裝之后你必要采取的幾個步驟,以保證一切運行良好。其中一個步驟修正了并行安裝造成的ASP.NET AJAX問題。

      代碼的動態(tài)編譯并執(zhí)行是一個.NET平臺提供給我們的很強大的工具用以靈活擴展(當然是面對內(nèi)部開發(fā)人員)復雜而無法估算的邏輯,并通過一些額外的代碼來擴展我們已有的應(yīng)用程序。這在很大程度上給我們提供了另外一種擴展的方式(當然這并不能算是嚴格意義上的擴展,但至少為我們提供了一種思路)。

      動態(tài)代碼執(zhí)行可以應(yīng)用在諸如模板生成,外加邏輯擴展等一些場合。一個簡單的例子,為了網(wǎng)站那的響應(yīng)速度,HTML靜態(tài)頁面往往是我們最好的選擇,但基于數(shù)據(jù)驅(qū)動的網(wǎng)站往往又很難用靜態(tài)頁面實現(xiàn),那么將動態(tài)頁面生成html的工作或許就是一個很好的應(yīng)用場合。另外,對于一些模板的套用,我們同樣可以用它來做。另外這本身也是插件編寫的方式。

      最基本的動態(tài)編譯

      .Net為我們提供了很強大的支持來實現(xiàn)這一切我們可以去做的基礎(chǔ),主要應(yīng)用的兩個命名空間是:System.CodeDom.Compiler和Microsoft.CSharp或Microsoft.VisualBasic。另外還需要用到反射來動態(tài)執(zhí)行你的代碼。動態(tài)編譯并執(zhí)行代碼的原理其實在于將提供的源代碼交予CSharpCodeProvider來執(zhí)行編譯(其實和CSC沒什么兩樣),如果沒有任何編譯錯誤,生成的IL代碼會被編譯成DLL存放于于內(nèi)存并加載在某個應(yīng)用程序域(默認為當前)內(nèi)并通過反射的方式來調(diào)用其某個方法或者觸發(fā)某個事件等。之所以說它是插件編寫的一種方式也正是因為與此,我們可以通過預(yù)先定義好的借口來組織和擴展我們的程序并將其交還給主程序去觸發(fā)。一個基本的動態(tài)編譯并執(zhí)行代碼的步驟包括:

      ·將要被編譯和執(zhí)行的代碼讀入并以字符串方式保存

      ·聲明CSharpCodeProvider對象實例

      ·調(diào)用CSharpCodeProvider實例的CompileAssemblyFromSource方法編譯

      ·用反射生成被生成對象的實例(Assembly.CreateInstance)

      ·調(diào)用其方法

      以下代碼片段包含了完整的編譯和執(zhí)行過程:

      需要解釋的是,這里我們在傳遞編譯參數(shù)時設(shè)置了GenerateInMemory為true,這表明生成的DLL會被加載在內(nèi)存中(隨后被默認引用入當前應(yīng)用程序域)。在調(diào)用GetTime方法時我們需要加入?yún)?shù),傳遞object類型的數(shù)組并通過Reflection的InvokeMember來調(diào)用。在創(chuàng)建生成的Assembly中的對象實例時,需要注意用到的命名空間是你輸入代碼的真實命名空間。以下是我們輸入的測試代碼(為了方便,所有的代碼都在外部輸入,動態(tài)執(zhí)行時不做調(diào)整):

      運行附件中提供的程序,可以很容易得到一下結(jié)果:

      改進的執(zhí)行過程
      現(xiàn)在一切看起來很好,我們可以編譯代碼并把代碼加載到當前應(yīng)用程序域中來參與我們的活動,但你是否想過去卸載掉這段程序呢?更好的去控制程序呢?另外,當你運行這個程序很多遍的時候,你會發(fā)現(xiàn)占用內(nèi)存很大,而且每次執(zhí)行都會增大內(nèi)存使用。是否需要來解決這個問題呢?當然需要,否則你會發(fā)現(xiàn)這個東西根本沒用,我需要執(zhí)行的一些大的應(yīng)用會讓我的服務(wù)器crzay,不堪重負而瘋掉的。

      要解決這個問題我們需要來了解一下應(yīng)用程序域。.NET Application Domain是.NET提供的運行和承載一個活動的進程(Process)的容器,它將這個進程運行所需的代碼和數(shù)據(jù),隔離到一個小的范圍內(nèi),稱為Application Domain。當一個應(yīng)用程序運行時,Application Domains將所有的程序集/組件集加載到當前的應(yīng)用程序域中,并根據(jù)需要來調(diào)用。而對于動態(tài)生成的代碼/程序集,我們看起來好像并沒有辦法去管理它。其實不然,我們可以用Application Domain提供的管理程序集的辦法來動態(tài)加載和移除Assemblies來達到我們的提高性能的目的。具體怎么做呢,在前邊的基礎(chǔ)上增加以下步驟:

      ·創(chuàng)建另外一個Application Domain

      ·動態(tài)創(chuàng)建(編譯)代碼并保存到磁盤

      ·創(chuàng)建一個公共的遠程調(diào)用接口

      ·創(chuàng)建遠程調(diào)用接口的實例。并通過這個接口來訪問其方法。

      換句話來講就是將對象加載到另外一個AppDomain中并通過遠程調(diào)用的方法來調(diào)用。所謂遠程調(diào)用其實也就是跨應(yīng)用程序域調(diào)用,所以這個對象(動態(tài)代碼)必須繼承于MarshalByRefObject類。為了復用,這個接口被單獨提到一個工程中,并提供一個工廠來簡化每次的調(diào)用操作:

      接下來在原來基礎(chǔ)上需要修改的是:

      ·將編譯成的DLL保存到磁盤中。

      ·創(chuàng)建另外的AppDomain。

      ·獲得IRemoteInterface接口的引用。(將生成的DLL加載到額外的AppDomain)

      ·調(diào)用InvokeMethod方法來遠程調(diào)用。

      ·可以通過AppDomain.Unload()方法卸載程序集。

      以下是完整的代碼,演示了如何應(yīng)用這一方案。

      對于客戶端的輸入程序,我們需要繼承于MarshalByRefObject類和IRemoteInterface接口,并添加對RemoteAccess程序集的引用。以下為輸入:

      這樣,你可以通過適時的編譯,加載和卸載程序集來保證你的程序始終處于一個可控消耗的過程,并且達到了動態(tài)編譯的目的,而且因為在不同的應(yīng)用程序域中,讓你的本身的程序更加安全和健壯。

      分享:揭秘ASP.NET實現(xiàn)頁面間值傳遞的幾種方法
      第一種方法: 通過URL鏈接地址傳遞 以下為引用的內(nèi)容: send.aspx: protected void Button1_Click(object sende

      來源:模板無憂//所屬分類:.Net教程/更新時間:2009-09-16
      相關(guān).Net教程