Winform程序經(jīng)常需要引用一些第三方控件,這些控件大多以DLL的形式提供。另外,一般USB橋芯片的官方提供.net操作類(lèi)庫(kù)也都是DLL形式提供的。因此一個(gè)稍大的項(xiàng)目中往往有一大堆的零散的DLL文件,而這些DLL文件在發(fā)布(deployment)時(shí)都要部署到exe的同一文件夾下。
有時(shí)候這種部署方式會(huì)帶來(lái)麻煩,特別是我們沒(méi)有專(zhuān)門(mén)打包而只是簡(jiǎn)單的把exe和dll放在一起發(fā)給用戶(hù)時(shí),容易因?yàn)橐粌蓚€(gè)DLL文件丟失造成exe不能運(yùn)行。因此我們希望能否將這些獨(dú)立零散的DLL直接嵌入exe內(nèi)部,這樣只發(fā)給用戶(hù)一個(gè)exe就行了。
.net中能做到這個(gè)效果的,我找到了兩種辦法:一個(gè)是用工具ILMerge,一個(gè)是把DLL作為嵌入式資源。
(1)、ILMerge。
這個(gè)工具是MS官方提供的,在 http://www.microsoft.com/download/en/details.aspx?displaylang=en&id=17630 可以下載得到。這個(gè)工具能夠把幾個(gè)可執(zhí)行文件(exe或者dll)打包集成進(jìn)一個(gè)可執(zhí)行文件中,具體使用方法網(wǎng)上很多,這里不再贅述。值得說(shuō)明的是,我嘗試寫(xiě)了一個(gè).bat批處理來(lái)merge,效果非常好。利用pause指令還能隨時(shí)暫停ILMerge運(yùn)行過(guò)程,可以看到merge失敗時(shí)是哪里的問(wèn)題。
(2)、嵌入DLL作為資源。
推薦使用這種方式。這個(gè)方法是CLR via C#的作者發(fā)明的(貌似,反正我是從他那里學(xué)的),原帖的地址是 http://blogs.msdn.com/b/microsoft_press/archive/2010/02/03/jeffrey-richter-excerpt-2-from-clr-via-c-third-edition.aspx 操作步驟這里說(shuō)的很清楚了,我不再羅嗦。只強(qiáng)調(diào)一下我在移植的時(shí)候遇到的兩個(gè)問(wèn)題。
1、AppDomain.CurrentDomain.AssemblyResolve += (sender, args) => {
String resourceName = "AssemblyLoadingAndReflection." +
new AssemblyName(args.Name).Name + ".dll";
using (var stream = Assembly.GetExecutingAssembly().GetManifestResourceStream(resourceName)) {
Byte[] assemblyData = new Byte[stream.Length];
stream.Read(assemblyData, 0, assemblyData.Length);
return Assembly.Load(assemblyData);
}
};
這段程序的位置非常重要。經(jīng)過(guò)反復(fù)嘗試,這段應(yīng)該放在Program.cs中Main方法里,Application.Run(new Form1());這句代碼之前才能有效。若放在Form1的構(gòu)造函數(shù)或者其他地方則都不行。另外,若程序使用了Splash screen,則這段代碼應(yīng)該放在splash screen代碼的OnCreateMainForm()方法體中。
2、這段話(huà)中有一句是要改的,不能原封不動(dòng)的移植。String resourceName = "AssemblyLoadingAndReflection." + ··· 這里的"AssemblyLoadingAndReflection."必須是你的嵌入Dll資源名稱(chēng)的DLL名之前的那部分。這里有一個(gè)竅門(mén)就是,可以用reflector打開(kāi)你的exe,看看Resources里你集成進(jìn)去的那些個(gè)DLL名稱(chēng)之前的部分是什么,就把那個(gè)東西寫(xiě)在這里就OK了。我一般是把DLL添加進(jìn)工程的Resources中的,這樣也省的工程管理器中占地方。
DLL和exe應(yīng)該分開(kāi)部署還是應(yīng)該集成一個(gè)exe?這個(gè)問(wèn)題比較復(fù)雜。個(gè)人感覺(jué),如果DLL很多而且都很小,那很合適嵌入exe方便部署。但是如果DLL很大那就沒(méi)必要了。
另外,兩個(gè)集成的方法中個(gè)人感覺(jué)第二個(gè)比較好一些。因?yàn)榈诙€(gè)并沒(méi)有破壞各個(gè)DLL,而只是將他們和exe打包進(jìn)一個(gè)assembly了。而ILMerge卻是將各個(gè)dll揉碎了注入exe中的,這對(duì)于很多第三方購(gòu)買(mǎi)的DLL來(lái)說(shuō)不合適使用。
還有一個(gè)問(wèn)題就是,ILMerge經(jīng)常會(huì)集成失敗,原因我就沒(méi)去分析了。但是第二種方式卻一定會(huì)成功,呵呵!
本站僅提供存儲(chǔ)服務(wù),所有內(nèi)容均由用戶(hù)發(fā)布,如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請(qǐng)
點(diǎn)擊舉報(bào)。