停止 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 对象后输入名称。
我尝试了以下...
从 Student 中删除默认构造函数。这有效,但这意味着我必须使用不同的构造函数并传入 string.Empty 作为学生姓名。这样做的主要问题是我们有一些其他通用代码来创建新对象,这些代码要求我们的 classes 具有默认构造函数。它也不是我们所有 classes 的解决方案。
将 IEnumerable<IStudent>
更改为 List<IStudent>
。这行得通,这也是我在代码中开始做的,但它随后强制使用 List<> 而不是通用的 IEnumerable<>
。只是不够灵活。
创建 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>
类型,这意味着它是只读的而不是意味着被干涉(即使它在运行时是一个数组)。
但如果您对问题的其他解决方案感兴趣,我可以推荐不同的方法:
- 你说你有一个 UI 并且
Student
的默认 ctor
在那里,以便用户界面可以处理这种类型。听起来你让你的用户界面干扰了你的对象模型的设计。我建议您尝试实现一些旨在分离 UI 和 BLL 的架构模式,例如 MVVM 或 MVC。例如,在 MVVM 的情况下,您可能有一个 StudentViewModel
用于 UI,它可以有一个默认的 ctor
,并且您可以仅使用参数化的 ctor
保持 Student 干净.
如果这不是选项(如你所说,它不是所有 类 的选项),你可以尝试在注册类型时指定构造函数(see the docs).在这种情况下,您可以手动注册非集合构造函数,Autofac 使用它:
builder.RegisterType<School>().UsingConstructor(typeof(string));
编辑:
或者,如果你真的热衷于完全禁用整个功能,我想你可以去硬核并开源它的 s**t :) 创建你自己的分支,找到 [=43= .cs 文件和 void RegisterDefaultAdapters(IComponentRegistry componentRegistry)
方法中,您可以看到正在添加的默认注册源。只需删除 componentRegistry.AddRegistrationSource(new CollectionRegistrationSource());
行(文件开发分支上的第 247 行),然后构建您自己的并使用它。不要忘记修改单元测试 ;)
我正在寻找一种方法来阻止 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 对象后输入名称。
我尝试了以下...
从 Student 中删除默认构造函数。这有效,但这意味着我必须使用不同的构造函数并传入 string.Empty 作为学生姓名。这样做的主要问题是我们有一些其他通用代码来创建新对象,这些代码要求我们的 classes 具有默认构造函数。它也不是我们所有 classes 的解决方案。
将
IEnumerable<IStudent>
更改为List<IStudent>
。这行得通,这也是我在代码中开始做的,但它随后强制使用 List<> 而不是通用的IEnumerable<>
。只是不够灵活。创建
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>
类型,这意味着它是只读的而不是意味着被干涉(即使它在运行时是一个数组)。
但如果您对问题的其他解决方案感兴趣,我可以推荐不同的方法:
- 你说你有一个 UI 并且
Student
的默认ctor
在那里,以便用户界面可以处理这种类型。听起来你让你的用户界面干扰了你的对象模型的设计。我建议您尝试实现一些旨在分离 UI 和 BLL 的架构模式,例如 MVVM 或 MVC。例如,在 MVVM 的情况下,您可能有一个StudentViewModel
用于 UI,它可以有一个默认的ctor
,并且您可以仅使用参数化的ctor
保持 Student 干净. 如果这不是选项(如你所说,它不是所有 类 的选项),你可以尝试在注册类型时指定构造函数(see the docs).在这种情况下,您可以手动注册非集合构造函数,Autofac 使用它:
builder.RegisterType<School>().UsingConstructor(typeof(string));
编辑:
或者,如果你真的热衷于完全禁用整个功能,我想你可以去硬核并开源它的 s**t :) 创建你自己的分支,找到 [=43= .cs 文件和 void RegisterDefaultAdapters(IComponentRegistry componentRegistry)
方法中,您可以看到正在添加的默认注册源。只需删除 componentRegistry.AddRegistrationSource(new CollectionRegistrationSource());
行(文件开发分支上的第 247 行),然后构建您自己的并使用它。不要忘记修改单元测试 ;)