首页 > 解决方案 > IoC - 使用注入的对象参数解析 WPF 窗口

问题描述

我第一次尝试使用 IoC 容器。一个程序有许多不同类型的窗口,这些窗口通常通过MainWindow. 但是,原则上它也可以在链条的下游。这些窗口通常需要注入一个或多个单例类,我们在这里命名它们IManager1/2/3。那么,我是否需要将它们冒泡IManagersMainWindow或者我可以在根目录(在此处App class)解决每个窗口?

无论哪种情况,这将如何完成?

我不需要使用 DryIoc,但这正是我选择测试的。

   public partial class App : Application
{
    DryIoc.Container container = new DryIoc.Container();

    private void Application_Startup(object sender, StartupEventArgs e)
    {
        RegisterIoc();

        var mainwindow = new MainWindow(); 
        mainwindow.Show();
    }

    public void RegisterIoc()
    {
        container.Register<IManager1, Manager1>(Reuse.Singleton);
        container.Register<IManager2, Manager2>(Reuse.Singleton);
        container.Register<IManager3, Manager3>(Reuse.Singleton);
    }
}

public MainWindow()
{
    public MainWindow()
    {
    }

    void OpenNewWindow2()
    {
        var w = new Window2(?, ?, ?);
        w.Show();
    }
}

public class Window2
{
    IManager1 man1;
    IManager2 man2;
    IManager3 man3;

    public Window2(IManager1 man1, IManager2 man2, IManager3 man3)
    {
        this.man1 = man1;
        this.man2 = man3;
        this.man3 = man3;
    }
}

标签: c#wpfdependency-injectioninversion-of-controldryioc

解决方案


考虑遵循显式依赖原则并Window2依赖MainWindow

例如

public class MainWindow {
    private readonly Func<Window2> window2;

    public MainWindow(Func<Window2> window2) {
        this.window2 = window2;
    }

    public void OpenNewWindow2() {
        var window = window2(); // invoke factory delegate
        window.Show();
    }
}

上面显示了使用 aFunc<T>作为工厂委托,它将充当所需依赖项的延迟解析的包装器。每次调用工厂委托时,它都会解析 的新实例,从而取消手动初始化它Window2的控制;MainWindow

App.Application_Startup用作组合根,您可以在其中向容器注册所有内容并解析您MainWindow的根对象

public partial class App : Application {
    DryIoc.Container container;

    private void Application_Startup(object sender, StartupEventArgs e) {
        container = RegisterIoc();    
        var mainwindow = container.Resolve<MainWindow>(); 
        mainwindow.Show();
    }

    private DryIoc.Container RegisterIoc() {
        var container = new DryIoc.Container();
        container.Register<MainWindow>();
        container.Register<Window2>();
        container.Register<IManager1, Manager1>(Reuse.Singleton);
        container.Register<IManager2, Manager2>(Reuse.Singleton);
        container.Register<IManager3, Manager3>(Reuse.Singleton);
        return container;
    }
}

请注意,要解析窗口,您还需要确保它们也已注册到容器中。


推荐阅读