Autofac 无法从 asp.net 5 中的程序集扫描中解析类型

Autofac not resolving types from assembly scan in asp.net 5

我们有一个 dnx46 web 项目,它在启动期间使用 autofac 程序集扫描来注册类型。我们的 project.json 依赖项包括:

"Autofac.Configuration": "4.0.0-rc1-268",
"Autofac.Extensions.DependencyInjection": "4.0.0-rc1-177",
"Autofac.Extras.CommonServiceLocator": "3.2.0",
"Microsoft.AspNet.Hosting": "1.0.0-rc1-final",
"Microsoft.AspNet.IISPlatformHandler": "1.0.0-rc1-final",
"Microsoft.AspNet.Mvc": "6.0.0-rc1-final",
"Microsoft.AspNet.Mvc.TagHelpers": "6.0.0-rc1-final",
"Microsoft.AspNet.Mvc.ViewFeatures": "6.0.0-rc1-final",
"Microsoft.AspNet.Server.Kestrel": "1.0.0-rc1-final",
"Microsoft.AspNet.Session": "1.0.0-rc1-final",
"Microsoft.AspNet.StaticFiles": "1.0.0-rc1-final",
"Microsoft.AspNet.Tooling.Razor": "1.0.0-rc1-final",
"Microsoft.Extensions.Configuration.EnvironmentVariables": "1.0.0-rc1-final",
"Microsoft.Extensions.Configuration.Json": "1.0.0-rc1-final",
"Microsoft.Extensions.Logging": "1.0.0-rc1-final",
"Microsoft.Extensions.Logging.Console": "1.0.0-rc1-final",
"Microsoft.Extensions.Logging.Debug": "1.0.0-rc1-final",
"Microsoft.VisualStudio.Web.BrowserLink.Loader": "14.0.0-rc1-final",
"Newtonsoft.Json": "8.0.3"

我们的 startup.cs ConfigureServices 方法如下所示:

public IServiceProvider ConfigureServices(IServiceCollection services)
{
    services.AddSingleton(serviceType => Configuration);
    services.AddInstance<Microsoft.Extensions.Configuration.IConfiguration>(Configuration);

    services.AddCaching();
    services.AddSession();
    services.AddMvc();

    var builder = new ContainerBuilder();

    var assemblies = Directory.GetFiles(<ourBinDirectoryPath>, "*.dll", SearchOption.TopDirectoryOnly).Select(Assembly.LoadFrom);

    foreach (var assembly in assemblies)
    {
        builder.RegisterAssemblyTypes(assembly).Where(t => t.Name.EndsWith("Service") || t.Name.EndsWith("Repository") || t.Name.EndsWith("DataContext")).AsSelf().AsImplementedInterfaces();
        builder.RegisterAssemblyTypes(assembly).Where(t => !t.Name.EndsWith("Service") && !t.Name.EndsWith("Repository") && !t.Name.EndsWith("DataContext")).AsSelf().AsImplementedInterfaces().InstancePerDependency();
    }

    builder.Populate(services);
    var container = builder.Build();

    return container.Resolve<IServiceProvider>();
}

我们的流程是我们构建了几个项目,将此项目需要的 dll 放入自定义 bin 文件夹中。在启动时,我们扫描这些程序集以使用 Autofac 注册类型。我们在概念验证项目中做到了这一点,并且没有出现任何问题。但是在我们的新项目中,我们有以下问题:

  1. 如果您尝试在控制器构造函数中解析服务,它 抛出错误指示 Microsoft 依赖注入不能 解决服务。不确定为什么不使用 Autofac 提供商在这里。
  2. 如果我们在启动时手动解析单个类型以及我们的程序集扫描,我们将在控制器构造中得到一个错误,指示 Autofac 无法解析服务(这里使用 Autofac 提供程序,但仍然找不到我们的类型).
  3. 如果我在 ConfigureServices 中检查 returns 之前的构建器和容器,它似乎已经注册了我们所有的类型。但是,如果我更改控制器以注入 IServiceProvider,然后尝试从中解析服务,它会准确地使用 Autofac 提供程序,但我总是返回 null(未找到服务)。

在尝试了很多事情之后,我将我们的概念验证项目复制到我们的新解决方案中(以防我们在设置新项目时遗漏了一些小设置或其他东西),但不良行为仍在继续。我开始认为我们的程序集引入了一些错误(因为这些是复制后唯一发生变化的东西),但 Autofac 在扫描期间不会抛出错误,而是不会正确使用它扫描的类型.

编辑 1

我创建并发布了一个 very simplified example,它重现了我正在 运行 遇到的问题。您应该能够将其下拉并 运行 以重现错误。您应该在主控制器构造函数中收到一个错误,指示服务解析失败。

注意:运行此示例时,它会执行一个 .bat 文件,将来自域项目的输出 dll 放入您的用户配置文件 .nuget 文件夹中。

编辑 2

根据 @Travis Illig 的建议将接口拆分到抽象文件夹中。一旦我这样做了,解决方案就开始工作了。我们在 "real" 项目中拆分了它,但它不起作用。仍在尝试找出原因。

编辑 3

终于让我们的项目重新开始工作了。问题最终有几件事:

  1. 我们的 Web 项目中列出了一个抽象项目依赖项,我们还从该程序集中加载和扫描 dll(双重加载)。 (我认为这是个大问题。)
  2. 我们的 NuGet 版本很糟糕,所以我认为它们加载不正确。我们使用的是 1.0.0-1,这是一种无效的预发布格式。将其更新为 1.0.0-b1,一切似乎开始变得更好了。

这让我花了更长的时间才弄清楚,我目前最大的问题是 Autofac 会无提示地失败。我真的希望它会爆炸并给我一些调试输出或表明存在问题的东西。相反,一切似乎都可以正常扫描和加载,但是当你去使用容器时,什么都解决不了。

程序集的加载顺序可能是某些类型由于未加载必备程序集而无法加载。

例如,假设您有包含接口 IComponent 的程序集 A。还假设您的程序集 B 包含实现 IComponent 的 class Component。如果先加载程序集 B,类型 Component 将无法加载,因为它实现的接口未定义。

在进行类型扫描时,Autofac 会尽量保证安全,如果遇到无法加载的类型,也不会爆炸。您可以在 scanning registration source 中看到它使用特殊的安全方法进行加载。

在您的尖峰代码中,我敢打赌要加载的程序集要少得多,所以您最终以正确的顺序加载了东西。快乐的意外。当你进入更大的项目时,更容易控制加载顺序并导致这样的问题。