如何在 Prism 应用程序中手动解析类型时获取正确的 INavigationService 实例

How to get the correct INavigationService instance on resolving a type manually in Prism application

似乎当我手动解析类型并在其中请求 INavigationService 时,注入的实例与其他地方使用的实例不同。

为了澄清我的用例,这里摘录了相关文件。正如您在解析类型 SampleProcess 时看到的那样,INavigationService 将被注入,但实例与我在 ProcessService 中获得的实例不同。 (顺便说一句,这是正确的实例,可用于导航。在 SampleProcess 中注入的实例不能用于导航。)

关于为什么会发生这种情况的任何想法,更重要的是我如何才能将 INavigationService 的正确实例注入到 SampleProcess 中。是的,我可以提供它,例如通过一个方法传递它,但这不是那么漂亮。

App.xaml.cs

protected override void RegisterTypes(IContainerRegistry containerRegistry)
{
    containerRegistry.RegisterSingleton<ProcessService>();
    containerRegistry.Register<Processes.SampleProcess>();
}

ProcessService.cs

public class ProcessService
{
    private readonly IContainer container;
    private readonly INavigationService navigationService;

    public ProcessService(IContainer container, INavigationService navigationService)
    {
        this.container = container;
        this.navigationService = navigationService;
    }

    public void ExecuteProcess(ProcessEnum processEnumValue)
    {
        Type processType = processEnumValue switch
        {
            ProcessEnum.SampleProcess => typeof(Processes.SampleProcess),
            _ => throw new NotImplementedException()
        };

        var process = App.Current.Container.Resolve(processType) as IProcess;

        bool test = process.CheckNavigationService(navigationService); // will return false
    }
}

SampleProcess.cs

public class SampleProcess : IProcess
{
    private readonly INavigationService navigationService;

    public SampleProcess(INavigationService navigationService)
    {
        this.navigationService = navigationService;
    }

    public bool CheckNavigationService(INavigationService navigationService)
    {
        return this.navigationService == navigationService;
    }
}

Prism 中的导航基于您从哪个页面导航。因此,NavigationService 是一种瞬态服务。它是专门为您的 ViewModel 构建的,以便为您提供的实例知道它需要从哪个页面导航。

您的问题有多种可能的解决方案。

  1. 不要将 INavigationService 注入到另一个服务中,因为它不应该以这种方式处理。

  2. 将 INavigationService 作为参数提供给需要使用它的任何服务

  3. 只需将您的 ProcessService 设为瞬态

  4. 更新到 Prism 8 并使您的流程服务成为范围服务,因为我们在 Prism 8 中进行了更改,每次创建新页面时都会创建一个新范围。这意味着任何需要访问 NavigationService 的 Scoped 或 Transient 服务都能够注入与注入 ViewModel

    完全相同的实例

我通过向从视图模型(实际上具有正确的 INavigationService 实例)调用的 ExecuteProcess 方法添加 INavigationService 参数来解决此问题。然后,在该方法中,我通过执行以下操作解决了该过程:

var process = container.Resolve(processType, new[] { navigationService }) as IProcess;

这样,正确的 INavigationService 实例在 IProcess 实例中可用。

正如@Dan Siegel 所说,这可以通过使用 Prism 8 并将 ProcessService 注册为范围服务来进一步改进,因为这将消除 INavigationService 参数的必要性 ExecuteProcess.