停止 Autofac 归档 IEnumerable<>

Stop Autofac from filing IEnumerable<>

我正在寻找一种方法来阻止 Autofac 在我的构造函数中自动填充 IEnumerable<>

例如,我有一个名为 School 的 class 具有以下构造函数

public School(string name) {...}
public School(string name, IEnumerable<IStudent> students) {...}

您可以使用第一个构造函数创建 School 对象,然后使用 AddStudent 方法将学生添加到学校。

您可以使用第二个构造函数创建一个学校,并在构造学校对象时传入要添加的学生集合。

如果我创建 Func<string, ISchool> schoolFactory,然后调用 schoolFactory.Invoke("Test"),Autofac 将使用第二个构造函数并自动创建一个新的 Student 对象。

IStudent 接口和 Student classes 已在 Autofac 中注册,并且 Student class 具有默认构造函数。默认构造函数在那里,因为当用户添加新学生时,它在我们的 UI 中使用。新 Student 以空白名称创建,用户可以在创建 Student 对象后输入名称。

我尝试了以下...

  1. 从 Student 中删除默认构造函数。这有效,但这意味着我必须使用不同的构造函数并传入 string.Empty 作为学生姓名。这样做的主要问题是我们有一些其他通用代码来创建新对象,这些代码要求我们的 classes 具有默认构造函数。它也不是我们所有 classes 的解决方案。

  2. IEnumerable<IStudent> 更改为 List<IStudent>。这行得通,这也是我在代码中开始做的,但它随后强制使用 List<> 而不是通用的 IEnumerable<>。只是不够灵活。

  3. 创建 Func<string, IEnumerable<IStudent>, ISchool> schoolFactory 并使用 schoolFactory.Invoke("Test", Enumerable.Empty<IStudent>)。这也有效。这是更多的代码,但我想它更明确。

有没有通用的方法可以阻止 Autofac 自动填充 IEnumerable?

问得好。简短的回答是:我认为这是不可能的。当涉及到软件工程时,我真的不喜欢说不,所以这里有一个更长的解释:

您在这里看到的是 Autofac 中所谓的 "implicit relationship type"。具体来说,the one that supports resolving collections. These are implemented in Autofac via registration sources。注册源是一种根据给定标准(在本例中为类型的集合方面)动态处理注册的方法。您可以在 container.ComponentRegistry.Sources 属性 的容器中检查这些来源;最后一个应该对此负责。所以为了禁用这个功能,你必须能够以某种方式从你的容器中删除这个注册源,但这里有一个问题:这个 属性 是 IEnumerable<IRegistrationSource> 类型,这意味着它是只读的而不是意味着被干涉(即使它在运行时是一个数组)。

但如果您对问题的其他解决方案感兴趣,我可以推荐不同的方法:

  1. 你说你有一个 UI 并且 Student 的默认 ctor 在那里,以便用户界面可以处理这种类型。听起来你让你的用户界面干扰了你的对象模型的设计。我建议您尝试实现一些旨在分离 UI 和 BLL 的架构模式,例如 MVVM 或 MVC。例如,在 MVVM 的情况下,您可能有一个 StudentViewModel 用于 UI,它可以有一个默认的 ctor,并且您可以仅使用参数化的 ctor 保持 Student 干净.
  2. 如果这不是选项(如你所说,它不是所有 类 的选项),你可以尝试在注册类型时指定构造函数(see the docs).在这种情况下,您可以手动注册非集合构造函数,Autofac 使用它:

    builder.RegisterType<School>().UsingConstructor(typeof(string));
    

编辑:

或者,如果你真的热衷于完全禁用整个功能,我想你可以去硬核并开源它的 s**t :) 创建你自己的分支,找到 [=43= .cs 文件和 void RegisterDefaultAdapters(IComponentRegistry componentRegistry) 方法中,您可以看到正在添加的默认注册源。只需删除 componentRegistry.AddRegistrationSource(new CollectionRegistrationSource()); 行(文件开发分支上的第 247 行),然后构建您自己的并使用它。不要忘记修改单元测试 ;)