用vs2017用C#建立WinForm小程序,程序内用到了连接MySql的MySql.Data.dll和处理Json串的Newtonsoft.Json.dll两个dll,纠结于仅仅小程序发布时就要附带这么多dll,着实令人着急,网上搜索发现解决方案,参考了文章
https://blog.csdn.net/lin381825673/article/details/39122257
https://blog.csdn.net/call_me_lzm/article/details/51501620
将dll嵌入exe,总结起来方案有:
- 使用ILMerge等工具将dll嵌入exe中
- 将dll内嵌于exe中,并在首次运行时将其释放出来
- 将dll内嵌于exe中,并在使用到dll时,将其加载在内存中
对于1,经过初步了解,应该是要生成exe后,通过该工具将dll嵌入。这样每次修改程序都要重复这个操作,感觉太繁琐,所以直接跳过,寻找别的方案(当然也可能是我不够了解带来的误解,如果有了解的同学欢迎指出);对于2,虽然可以满足分发的时候只发布一个exe文件,但一旦运行起来又甩出一堆dll,看着还是不爽;3就是我采纳的方案,下面细说。
一、仍然需要将两个dll引入项目(否则代码编译不通过),dll文件也是必须的(运行时需要调用)。(这一步可能已经做过了,那么看下面两步)
二、双击Resources.resx,选择添加现有文件,把所需DLL添加进来
这时可以看到多了:
三、添加AssemblyResolve处理函数
System.Reflection.Assembly CurrentDomain_AssemblyResolve(object sender, ResolveEventArgs args)
{
string dllName = args.Name.Contains(",") ? args.Name.Substring(0, args.Name.IndexOf(',')) : args.Name.Replace(".dll", "");
dllName = dllName.Replace(".", "_");
if (dllName.EndsWith("_resources")) return null;
System.Resources.ResourceManager rm = new System.Resources.ResourceManager(GetType().Namespace + ".Properties.Resources", System.Reflection.Assembly.GetExecutingAssembly());
byte[] bytes = (byte[])rm.GetObject(dllName);
return System.Reflection.Assembly.Load(bytes);
}
public Form1()//看清楚这是窗体本来的初始化函数
{
//在InitializeComponent()之前调用
AppDomain.CurrentDomain.AssemblyResolve += new ResolveEventHandler(CurrentDomain_AssemblyResolve);
InitializeComponent();
}
可以在CurrentDomain_AssemblyResolve中设置断点查看,会发现当程序中首次调用到MySql.Data.dll或者Newtonsoft.Json.dll的方法时会回调该函数。并且生成exe文件后,删除dll文件单独运行exe,也不会再报找不到dll的错误了。一旦dll被加载到内存中,就会一直存在。所以只有首次找不到dll时会去加载,其他时候就不用了。
在试验过程中遇到了问题,当第二步添加资源时只把Newtonsoft.Json.dll添加进去,而不管MySql.Data.dll时,程序发布时没有dll仍然会报错,调试了好久,把所有dll全部加入资源,才成功了!