c#項目將dll打包到exe中的步驟
意圖:
想將項目用到的兩個dll庫文件(CryptEnDe.dll和ICSharpCode.SharpZipLib.dll)一同編譯進exe中,并編譯后僅一個exe程序就可以獨立運行不再需要其它文件。
實現(xiàn):
1、將兩個dll庫文件作為資源文件添加進項目中;
2、添加功能代碼
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Reflection;
using System.IO;
namespace AutoUpdateServer.Core
{
/// <summary> 載入資源中的動態(tài)鏈接庫(dll)文件
/// </summary>
static class LoadResourceDll
{
static Dictionary<string, Assembly> Dlls = new Dictionary<string, Assembly>();
static Dictionary<string, object> Assemblies = new Dictionary<string, object>();
static Assembly AssemblyResolve(object sender, ResolveEventArgs args)
{
//程序集
Assembly ass;
//獲取加載失敗的程序集的全名
var assName = new AssemblyName(args.Name).FullName;
//判斷Dlls集合中是否有已加載的同名程序集
if (Dlls.TryGetValue(assName, out ass) && ass != null)
{
Dlls[assName] = null;//如果有則置空并返回
return ass;
}
else
{
throw new DllNotFoundException(assName);//否則拋出加載失敗的異常
}
}
/// <summary> 注冊資源中的dll
/// </summary>
public static void RegistDLL()
{
//獲取調(diào)用者的程序集
var ass = new StackTrace(0).GetFrame(1).GetMethod().Module.Assembly;
//判斷程序集是否已經(jīng)處理
if (Assemblies.ContainsKey(ass.FullName))
{
return;
}
//程序集加入已處理集合
Assemblies.Add(ass.FullName, null);
//綁定程序集加載失敗事件(這里我測試了,就算重復(fù)綁也是沒關(guān)系的)
AppDomain.CurrentDomain.AssemblyResolve += AssemblyResolve;
//獲取所有資源文件文件名
var res = ass.GetManifestResourceNames();
foreach (var r in res)
{
//如果是dll,則加載
if (r.EndsWith(".dll", StringComparison.OrdinalIgnoreCase))
{
try
{
if (r.Contains("CryptEnDe.dll")
|| r.Contains("CryptEnDe_d.dll"))
{
ExtractResourceToFile(r, PathUtils.GetUpdateDllPath() + @"/" + r.Substring(r.IndexOf('.') + 1));
}
var s = ass.GetManifestResourceStream(r);
var bts = new byte[s.Length];
s.Read(bts, 0, (int)s.Length);
var da = Assembly.Load(bts);
//判斷是否已經(jīng)加載
if (Dlls.ContainsKey(da.FullName))
{
continue;
}
Dlls[da.FullName] = da;
}
catch(Exception e)
{
//加載失敗就算了...
}
}
}
}
private static void ExtractResourceToFile(string resourceName, string filename)
{
//if (!System.IO.File.Exists(filename))
{
using (System.IO.Stream s = System.Reflection.Assembly.GetExecutingAssembly().GetManifestResourceStream(resourceName))
{
using (System.IO.FileStream fs = new System.IO.FileStream(filename, System.IO.FileMode.Create, FileAccess.ReadWrite, FileShare.ReadWrite))
{
byte[] b = new byte[s.Length];
s.Read(b, 0, b.Length);
fs.Write(b, 0, b.Length);
}
}
}
}
}
}
其中PathUtils.GetUpdateDllPath()函數(shù)為獲取dll釋放的路徑,根據(jù)自己需要指定路徑
public static string GetUpdateDllPath()
{
string strPath = @"C:/LTShiyi/cache/updatedll";
if (!Directory.Exists(strPath))
{
Directory.CreateDirectory(strPath);
}
return strPath;
}
3、在程序入口Program類中調(diào)用上面的接口函數(shù)
static Program()
{
AutoUpdateServer.Core.LoadResourceDll.RegistDLL();
}
4、編譯即可。
以上就是c#項目將dll打包到exe中的步驟的詳細內(nèi)容,更多關(guān)于c# dll打包到exe的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
C#?TaskScheduler任務(wù)調(diào)度器的實現(xiàn)
本文主要介紹了C#?TaskScheduler任務(wù)調(diào)度器的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧<BR>2023-05-05
c#不使用windows api函數(shù)打開我的電腦和獲取電腦驅(qū)動器信息
這篇文章主要介紹了c#不使用windows api函數(shù)打開我的電腦和電腦驅(qū)動器信息的方法,大家參考使用2013-12-12
Entity?Framework使用ObjectContext類
這篇文章介紹了Entity?Framework使用ObjectContext類的方法,文中通過示例代碼介紹的非常詳細。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-06-06
Winform開發(fā)中使用下拉列表展示字典數(shù)據(jù)的幾種方式
這篇文章介紹了Winform開發(fā)中使用下拉列表展示字典數(shù)據(jù)的幾種方式,文中通過示例代碼介紹的非常詳細。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-09-09

