C#代碼動態(tài)編譯、動態(tài)執(zhí)行、動態(tài)調(diào)試.doc_第1頁
C#代碼動態(tài)編譯、動態(tài)執(zhí)行、動態(tài)調(diào)試.doc_第2頁
C#代碼動態(tài)編譯、動態(tài)執(zhí)行、動態(tài)調(diào)試.doc_第3頁
C#代碼動態(tài)編譯、動態(tài)執(zhí)行、動態(tài)調(diào)試.doc_第4頁
C#代碼動態(tài)編譯、動態(tài)執(zhí)行、動態(tài)調(diào)試.doc_第5頁
免費預(yù)覽已結(jié)束,剩余1頁可下載查看

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報或認(rèn)領(lǐng)

文檔簡介

C#代碼動態(tài)編譯、動態(tài)執(zhí)行、動態(tài)調(diào)試前幾天看到一篇關(guān)于.net動態(tài)編譯的文章 .NET中的動態(tài)編譯 ,很受啟發(fā)。在此基礎(chǔ)上我做了一些封裝,為使調(diào)用更加簡單,并增加了對動態(tài)代碼調(diào)試的支持,相同代碼只編譯一次的支持,代碼改動自動重新編譯,代碼引用文件的自動加載和手工加載等功能。 如上圖,我封裝的類CSharpProvider很簡單,下面說明一下一些公共成員的用法。 公共屬性 AssemblyFileName:這個屬性指定動態(tài)編譯后生成的配件名稱。 CompilerParameters:這個屬性指定編譯的參數(shù) References:這個屬性指定被編譯代碼中的引用。調(diào)用者只要調(diào)用References.Add(xxx.dll),就可以加入自己的引用,對于System命名空間的所有引用,不需要手工加入,該類會自動加載。對于用戶自己的組件,如果不手工指定引用文件,該類會自動根據(jù)名字空間名進(jìn)行猜測。 SourceCodeFileEncoding:如果以文件形式編譯,指定文件的編碼類型。 公共方法 public bool Compile(string code) 輸入代碼字符串,并編譯 public bool CompileFromFile(string sourceCodeFileName) 編譯輸入的代碼文件 public object CreateInstance(string code, string typeFullName) 創(chuàng)建類的實例 如下面代碼,可以輸入 CreateInstance(code, MyInterface.IHelloWorld),也可以輸入CreateInstance(code, HelloWorld),程序會根據(jù)類型名稱來自動找到符合條件的類并實例化。如果代碼中有多個指定類型的類,將實例化第一個。usingSystem;usingMyInterface;SerializablepublicclassHelloWorld:MarshalByRefObject,IHelloWorldpublicstringSay()returnHi; 這里需要特別指出的是由于用到了AppDomain的遠(yuǎn)程調(diào)用,所有的動態(tài)加載的代碼必須繼承自MarshallByRefObject如果僅僅聲明為Serializable 雖然也可以執(zhí)行,但主應(yīng)用程序域會記錄下子應(yīng)用程序域的一個引用,這樣導(dǎo)致子應(yīng)用程序域卸載后,依然無法完全釋放內(nèi)存,從而內(nèi)存泄漏。所以這個很關(guān)鍵,一定要注意。 public object CreateInstanceFromFile(string fileName, string typeFullName) 從文件創(chuàng)建動態(tài)實例 下面再談?wù)剬討B(tài)代碼的調(diào)試 動態(tài)創(chuàng)建的代碼如果不能調(diào)試,就像一個黑盒子,對系統(tǒng)的可維護(hù)性有較大破壞。未來實現(xiàn)這個功能,我們需要做以下工作,第一、編譯時要生成調(diào)試信息,這個可以通過設(shè)置 CompilerParameters.IncludeDebugInformation = true;來實現(xiàn)第二、我們必須告訴調(diào)試器源碼對應(yīng)的位置,對于從文件編譯的情況,源碼文件位置會被自動寫入調(diào)試信息文件 *.pdb中,而對于從內(nèi)存編譯的情況,我還沒有找到指定的方法,如果哪位朋友知道,還望賜教。所以目前如果要調(diào)試動態(tài)代碼,必須從文件編譯,也就是調(diào)用CompileFromFile,CreateInstanceFromFile。第三、我們需要在代碼中設(shè)置一個斷點,這個可以在代碼中加入 System.Diagnostics.Debugger.Break(); 來解決。如下圖所示,動態(tài)代碼現(xiàn)在可以調(diào)試了。應(yīng)用程序域為了避免內(nèi)存泄漏,本程序封裝了對應(yīng)用程序域的使用,調(diào)用者基本不需要關(guān)心應(yīng)用程序域的調(diào)用和卸載過程。本程序在重新編譯或者對象銷毀時會自動卸載應(yīng)用程序域,從而釋放內(nèi)存。由于做這個程序是在應(yīng)用程序域上遇到了很多麻煩,所以感覺還是有必要簡單講一下應(yīng)用程序域。如上圖所示,應(yīng)用程序與實際上有點像一個單獨的進(jìn)程,但這個進(jìn)程是運行在當(dāng)前進(jìn)程里面的,當(dāng)然這個比喻不夠貼切。對應(yīng)用程序域的調(diào)用有點類似進(jìn)程間采用 Remoting 方式的對象調(diào)用,也就是說默認(rèn)應(yīng)用程序域要調(diào)用其他應(yīng)用程序域中的對象,必須采用遠(yuǎn)程調(diào)用的方法,而不能直接調(diào)用,如果直接調(diào)用,默認(rèn)應(yīng)用程序域就會記錄這個被調(diào)用的應(yīng)用程序域的一個內(nèi)存引用,即使這個應(yīng)用程序域執(zhí)行了Unload 方法卸載后,內(nèi)存依然無法釋放,這也是我一開始操作應(yīng)用程序域遇到的最大困擾。另外所有暴露在兩個應(yīng)用程序域之間的類必須從MarshalByRefObject基礎(chǔ),這點非常重要,否則將導(dǎo)致內(nèi)存無法釋放。

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

評論

0/150

提交評論