如何将 C++ windows dll 合并到 C# 应用程序 exe 中?
Posted
技术标签:
【中文标题】如何将 C++ windows dll 合并到 C# 应用程序 exe 中?【英文标题】:How can a C++ windows dll be merged into a C# application exe? 【发布时间】:2010-09-09 11:49:14 【问题描述】:我有一个 Windows C# 程序,它使用 C++ dll 进行数据 i/o。我的目标是将应用程序部署为单个 EXE。
创建这样一个可执行文件的步骤是什么?
【问题讨论】:
您应该使用 BoxedAppPacker 或 BoxedApp SDK。它必须有帮助。 重复***.com/questions/666799/… 【参考方案1】:托管和非托管代码的单一程序集部署 2007 年 2 月 4 日,星期日
.NET 开发人员喜欢 XCOPY 部署。他们喜欢单一的装配组件。至少我总是觉得有点不安,如果我必须使用某个组件并且需要记住一个文件列表,还需要包含在该组件的主程序集中。因此,当我最近不得不开发一个托管代码组件并不得不使用来自 C DLL 的一些非托管代码来扩充它时(感谢 Marcus Heege 帮助我解决这个问题!),我想到了如何更轻松地部署这两个 DLL .如果这只是两个程序集,我可以使用 ILmerge 将它们打包到一个文件中。但这不适用于具有托管和非托管 DLL 的混合代码组件。
所以这是我想出的解决方案:
我将我想要与组件的主程序集一起部署的任何 DLL 作为嵌入式资源包含在内。 然后我设置了一个类构造函数来提取这些 DLL,如下所示。 ctor 类在每个 AppDomain 中只被调用一次,所以我认为它的开销可以忽略不计。
namespace MyLib
public class MyClass
static MyClass()
ResourceExtractor.ExtractResourceToFile("MyLib.ManagedService.dll", "managedservice.dll");
ResourceExtractor.ExtractResourceToFile("MyLib.UnmanagedService.dll", "unmanagedservice.dll");
...
在此示例中,我包含了两个 DLL 作为资源,一个是非托管代码 DLL,一个是托管代码 DLL(仅用于演示目的),以展示该技术如何适用于这两种代码。
将 DLL 提取到自己的文件中的代码很简单:
public static class ResourceExtractor
public 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))
byte[] b = new byte[s.Length];
s.Read(b, 0, b.Length);
fs.Write(b, 0, b.Length);
使用这样的托管代码程序集与往常一样 - 几乎。您在组件的主项目(此处:MyLib)中引用它(此处:ManagedService.dll),但将 Copy Local 属性设置为 false。此外,您将程序集链接为现有项目并将构建操作设置为嵌入式资源。
对于非托管代码(此处:UnmanagedService.dll),您只需将 DLL 作为现有项链接,并将构建操作设置为嵌入式资源。要访问其功能,请照常使用 DllImport 属性,例如
[DllImport("unmanagedservice.dll")] public extern static int Add(int a, int b);
就是这样!一旦您使用静态 ctor 创建了该类的第一个实例,嵌入式 DLL 就会被提取到它们自己的文件中并准备好使用,就像您将它们部署为单独的文件一样。只要您对执行目录具有写入权限,这对您来说应该可以正常工作。至少对于原型代码,我认为这种单程序集部署方式是相当方便的。
享受吧!
http://weblogs.asp.net/ralfw/archive/2007/02/04/single-assembly-deployment-of-managed-and-unmanaged-code.aspx
【讨论】:
链接已失效。这就是为什么您应该始终 quote the most relevant parts of the linked page 而不是发布仅链接的答案。【参考方案2】:试试boxedapp;它允许从内存中加载所有 DLL。此外,您甚至可以嵌入 .net 运行时。很好地创建一个真正独立的应用程序...
【讨论】:
【参考方案3】:使用 Fody.Costura nuget
-
打开您的解决方案 -> 项目 -> 管理 Nuget 包
搜索 Fody.Costura
编译您的项目。
就是这样!
来源: http://www.manuelmeyer.net/2016/01/net-power-tip-10-merging-assemblies/
【讨论】:
【参考方案4】:您尝试过 ILMerge 吗? http://research.microsoft.com/~mbarnett/ILMerge.aspx
ILMerge 是一个实用程序,可用于将多个 .NET 程序集合并为一个程序集。它可从 Microsoft .NET Framework 开发人员中心的工具和实用程序页面免费使用。
如果您使用 /clr
标志(全部或部分 C++/CLI)构建 C++ DLL,那么它应该可以工作:
ilmerge /out:Composite.exe MyMainApp.exe Utility.dll
但是,它不适用于普通(本机)Windows DLL。
【讨论】:
很确定它需要 DLL 上的/clr:pure
或更高版本。【参考方案5】:
只需在 Visual Studio 中右键单击您的项目,选择项目属性 -> 资源 -> 添加资源 -> 添加现有文件... 并将下面的代码包含到您的 App.xaml.cs 或等效代码中。
public App()
AppDomain.CurrentDomain.AssemblyResolve +=new ResolveEventHandler(CurrentDomain_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);
这是我的原始博客文章: http://codeblog.larsholm.net/2011/06/embed-dlls-easily-in-a-net-assembly/
【讨论】:
请不要多次发布完全相同的帖子,特别是如果您要发布非常老的问题。如果与上一篇文章完全相同的答案是合适的,请将问题标记为重复。如果您继续发布带有相同文本的博客链接,您可能会吸引一些垃圾邮件标记。【参考方案6】:Thinstall 是一种解决方案。对于本机 Windows 应用程序,我建议将 DLL 嵌入为二进制资源对象,然后在运行时在需要它之前将其提取出来。
【讨论】:
Thinstall 非常昂贵,并且在某些 Windows 安装和第 3 方 shell 扩展方面遇到问题。我们将它用于非 .Net 代码的成功有限,但发现 ClickOnce 和 MSI 为 .Net 安装了最好的。 如果您主要是在寻找安装,我会推荐 Inno Setup,几年前我们为此放弃了 InstallShield,并且从未回头。 innosetup.com/isinfo.php【参考方案7】:Smart Assembly 可以做到这一点,甚至更多。如果您的 dll 具有非托管代码,它不会让您将 dll 合并到单个程序集中,而是可以将所需的依赖项作为资源嵌入到您的主 exe 中。它的另一面,它不是免费的。
您可以手动执行此操作,方法是将 dll 嵌入到您的资源中,然后依赖 AppDomain 的 Assembly ResolveHandler
。当谈到混合模式 dll 时,我发现 ResolveHandler
方法的许多变体和风格对我不起作用(所有这些都将 dll 字节读取到内存并从中读取)。他们都为托管 dll 工作。这对我有用:
static void Main()
AppDomain.CurrentDomain.AssemblyResolve += (sender, args) =>
string assemblyName = new AssemblyName(args.Name).Name;
if (assemblyName.EndsWith(".resources"))
return null;
string dllName = assemblyName + ".dll";
string dllFullPath = Path.Combine(GetMyApplicationSpecificPath(), dllName);
using (Stream s = Assembly.GetEntryAssembly().GetManifestResourceStream(typeof(Program).Namespace + ".Resources." + dllName))
byte[] data = new byte[stream.Length];
s.Read(data, 0, data.Length);
//or just byte[] data = new BinaryReader(s).ReadBytes((int)s.Length);
File.WriteAllBytes(dllFullPath, data);
return Assembly.LoadFrom(dllFullPath);
;
这里的关键是将字节写入文件并从其位置加载。为了避免先有鸡还是先有蛋的问题,您必须确保在访问程序集之前声明处理程序,并且不要在加载(程序集解析)部分中访问程序集成员(或实例化任何必须处理程序集的东西)。还要注意确保GetMyApplicationSpecificPath()
不是任何临时目录,因为临时文件可能会被其他程序或您自己尝试擦除(并不是说它会在您的程序访问 dll 时被删除,但至少它是一个麻烦。 AppData 位置很好)。另请注意,您每次都必须写入字节,您不能从位置加载,因为 dll 已经存在于那里。
如果程序集完全不受管理,您可以查看link 或this 了解如何加载此类 dll。
【讨论】:
【参考方案8】:Xenocode 的 PostBuild 可以将托管和非托管打包成一个 exe。
【讨论】:
以上是关于如何将 C++ windows dll 合并到 C# 应用程序 exe 中?的主要内容,如果未能解决你的问题,请参考以下文章
如何将dll文件移动到C:\WINDOWS\assembly文件夹
如何将结构从 C++ 应用程序传递到 C++ Win32 DLL?