c# - 将 IoC/DI 容器与运行时相关的构造函数参数一起使用

标签 c# dependency-injection structuremap

我正在转换我的代码以使用带 StructureMap 的 IoC 容器。试图让我的头脑围绕着事情,我觉得它开始“点击”​​,我可以看到它对后端有多么有意义。

但是,我一直在努力,我发现了一些我不确定如何让它发挥作用的情况。具体来说,我的原始构造函数使用一个实际上不是依赖项的参数做了一些重要的事情,或者在运行时会改变的事情。

假设我从这个(预 IoC 容器)开始,我在其中使用构造函数传递我的依赖项,但也向它发送一个运行时相关的 ImportantObject:

IMyPageViewModel myPageViewModel = new MyPageViewModel(importantObject, dialogManager, pageDisplay, viewModelProvider)

这里它正在构建:

public MyPageViewModel(ImportantObject importantObject, IDialogManager dialogManager,IPageDisplay pageDisplay, IViewModelProvider viewModelProvider)
{
    this.dialogManager = dialogManager;
    this.pageDisplay = pageDisplay;
    this.viewModelProvider = viewModelProvider;

    importantObject.DoThatImportantThing();
}

现在,我正在迁移以使用 IoC 容器,起初我认为我应该这样做:

//I need to create an instance to use, so I use my IoC container:
IMyPageViewModel myPageViewModel = container.GetInstance<IMyPageViewModel>();

然后让它解决它的依赖关系,但是 importantObject 是在运行时设置的。我无法将其注册为依赖项:

public MyPageViewModel(IDialogManager dialogManager,IPageDisplay pageDisplay, IViewModelProvider viewModelProvider, IContainer container)
{
    this.dialogManager = dialogManager;
    this.pageDisplay = pageDisplay;
    this.viewModelProvider = viewModelProvider;

    //however, here I have no access to the important object that I previously passed in my constructor
    importantObject.DoThatImportantThing(); //obviously error
}

我想也许我应该使用“new”创建并传递 IoC 容器:

IMyPageViewModel myPageViewModel = new MyPageViewModel(importantObject, container)

然后让它在构造函数中解析它的依赖关系:

public MyPageViewModel(ImportantObject importantObject, IContainer container)
{
    this.dialogManager = container.GetInstance<IDialogManager>();
    this.pageDisplay = container.GetInstance<IPageDisplay>();
    this.viewModelProvider = container.GetInstance<IViewModelProvider>();

    importantObject.DoThatImportantThing();
}

但这让我觉得这不是一个好主意,具体来说,我不能用测试寄存器运行它,也不能让它为单元测试创​​建一个虚拟/ stub “MyPageViewModel”。

我唯一能想到的另一件事是从构造函数中删除所有逻辑并将其放入初始化方法或属性 setter 中。但是,这意味着我必须确保始终在使用前调用初始化,它会隐藏错误/问题。

这些选项中的任何一个是否合理,我应该如何管理在具有依赖注入(inject)的构造函数中传递运行时依赖对象?

我试图远离静态工厂,因为我读过很多关于它们是反模式/不良做法的文章。

编辑:为了回应 Bruno Garcia 的回答,我决定使用工厂类型模式来保存容器并像这样处理对象创建:

class PageProvider : IPageProvider
{
    public MyPageViewModel GetMyPage(ImportantObject importantObject)
    {
        //might just get, if it's a single only instance
        return MyPageViewModel(ImportantObject importantObject,
                               container.GetInstance<IDialogManager>(),
                               container.GetInstance<IPageDisplay>(),
                               container.GetInstance<IViewModelProvider>())
    }
}

最佳答案

结构图 supports passing arguments解决。这可以帮助您通过 ImportantObject到您正在解析的服务。

值得注意的是,如果您传递容器,事情很快就会变得非常困惑。避免将其用作 Service Locator .

理想情况下,您会使用容器来解析入口点(例如:Controller、Consumer worker),从那时起,就不再直接使用容器了。如果您需要控制构造函数中依赖项的生命周期,有多种方法可以做到这一点,例如: 拍摄FactoryFunc<> .

我建议您仔细阅读您要使用的容器的文档,以了解谁控制对象的生命周期(如果组件实现了 IDisposable,谁将处置它?)。何时创建/处置生命周期范围?

IoC 容器很棒,但如果您不仔细理解生命周期所有权的概念,很容易发现自己在解决内存泄漏问题。

关于c# - 将 IoC/DI 容器与运行时相关的构造函数参数一起使用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37254794/

相关文章:

c# - 使用数组填充列表框?

c# - MEF - 将默认创建策略更改为 NonShared

c# - 如何在 JsonConverter 中注入(inject)/访问 HttpContext?

c# - 为实现接口(interface)的所有类型注册通用工厂

c# - StructureMap 中的命名实例和默认实例?

c# - 如何使用 StructureMap 通过代码定义默认构造函数?

C# 8.0 基于输入类型的 switch 表达式

c# - 获取MySQL多行数据并显示在单独的文本框中

c# - Unity - 如何在解析/实例化对象时收到通知?

.net - 使用 Structuremap 更改范围的多个构造函数?