尝试将 Autofac 作为嵌入程序集加载时出现 FileNotFoundException
本文关键字:加载 程序集 FileNotFoundException Autofac | 更新日期: 2023-09-27 18:31:11
以前版本的Autofac可以工作,但由于他们改用可移植类库,因此无法加载。
我尝试应用此处列出的修复程序(KB2468871),但它告诉我不需要它。
当我将 Autofac.dll 文件移动到与可执行文件相同的位置时,错误消失了。 当它从外部DLL加载它时,它会加载正常。
为什么它不能作为嵌入式 DLL 工作?
以下是例外:
System.IO.FileNotFoundException: Could not load file or assembly 'System.Core, Version=2.0.5.0, Culture=neutral, PublicKeyToken=7cec85d7bea7798e, Retargetable=Yes' or one of its dependencies. The system cannot find the file specified.
Stack trace:
at Autofac.Core.Registration.ComponentRegistration..ctor(Guid id, IInstanceActivator activator, IComponentLifetime lifetime, InstanceSharing sharing, InstanceOwnership ownership, IEnumerable`1 services, IDictionary`2 metadata)
at Autofac.Core.Container..ctor()
at Autofac.ContainerBuilder.Build(ContainerBuildOptions options)
at MyApp.Configuration.Bootstrapper.Run(String[] args) in c:'Dev'MyApp'App'Configuration'Bootstrapper.cs:line 25
at MyApp.Configuration.EntryPoint.Main(String[] args) in c:'Dev'MyApp'App'Configuration'EntryPoint.cs:line 22
如果有帮助,下面是将 DLL 嵌入可执行文件的 .csproj 文件的一部分:
<Target Name="AfterResolveReferences">
<ItemGroup>
<EmbeddedResource Include="@(ReferenceCopyLocalPaths)" Condition="'%(ReferenceCopyLocalPaths.Extension)' == '.dll'">
<LogicalName>%(ReferenceCopyLocalPaths.DestinationSubDirectory)%(ReferenceCopyLocalPaths.Filename)%(ReferenceCopyLocalPaths.Extension)</LogicalName>
</EmbeddedResource>
</ItemGroup>
。下面是入口点类:
internal static class EntryPoint
{
[STAThread]
private static void Main(params string[] args)
{
AppDomain.CurrentDomain.AssemblyResolve += (sender, e) => loadEmbeddedAssembly(e.Name);
Bootstrapper.Run(args); // must call separate class when using embedded assemblies
}
private static Assembly loadEmbeddedAssembly(string name)
{
var container = Assembly.GetExecutingAssembly();
var path = new AssemblyName(name).Name + ".dll";
using (var stream = container.GetManifestResourceStream(path))
{
if (stream == null)
{
return null;
}
var bytes = new byte[stream.Length];
stream.Read(bytes, 0, bytes.Length);
return Assembly.Load(bytes);
}
}
}
我可以通过在面向 Silverlight 并使用 System.Core 类型的 PCL 库上使用 Assembly.Load(byte[]) 轻松重现您的问题,这个问题并非特定于 Autofac。 这是一种相当邪恶的加载程序集的方式,与Assembly.LoadFile()一样糟糕。 没有加载上下文是DLL Hell的秘诀。
CLR 对这些 PCL 库引用有一项不令人羡慕的工作,它需要神奇地将可重定向的程序集引用映射到真实的程序集引用。 换句话说,2.0.5.0 引用需要映射到运行时版本的正确引用,在您的情况下为 4.0.0.0。 只有当它很清楚实际的运行时版本是什么时,它才能做到这一点,没有加载上下文会使这变得困难。 显然,它不会尝试这样做,它会再次触发 AssemblyResolve 事件以获取 2.0.5.0 引用。
这是解决方案,事后看来非常简单。 只需截获可重定目标程序集引用的解析请求,并使用 Assembly.Load() 让 CLR 从 AppDomain 的加载上下文中对其进行排序。 更改 AssemblyResolve 事件处理程序,如下所示:
private static Assembly loadEmbeddedAssembly(string name)
{
if (name.EndsWith("Retargetable=Yes")) {
return Assembly.Load(new AssemblyName(name));
}
// Rest of your code
//...
}
这在我的测试应用程序中运行良好,我相信它也会解决您的 Autofac 问题。
也许问题在于您尝试加载嵌入式程序集的方式。
我对你怎么做
知之甚少但我知道以下来自代码项目的示例对我来说很好用,所以也许它会帮助你:-)
从嵌入的资源加载 DLL
(如果我误解了您的问题,请告诉我,我将删除我的答案)