RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / 问题 / 1184712
Accepted
aepot
aepot
Asked:2020-10-01 16:19:42 +0000 UTC2020-10-01 16:19:42 +0000 UTC 2020-10-01 16:19:42 +0000 UTC

用于傻瓜的 MVVM 中的 IoC

  • 772

我最终决定学习如何在 MVVM 框架内使应用程序界面可扩展为多个 View / ViewModel。我遇到了Inversion of Control + Depency Injection 模板,上面有很多信息,很多人建议用它来组织应用程序的逻辑架构。此外,它还提供了组织诸如单例存储等便利的机会,这正是我所需要的。

问题是,在作者推荐的文章中,这就是全部,几乎没有人在 WPF + MVVM 中“为傻瓜”手动显示 IoC 实现。

如果不了解它是如何工作的,我就不能在我的项目中使用“黑匣子”,我通常不喜欢大的、可怕的和无法理解的东西,不清楚它是如何工作的,我总是试图挖掘这个非常本质。我决定对一项实际任务进行分析 - 制作带有ListBox页面选择的多页面界面。

从两个接口开始

public interface IView
{
    string Title { get; set; }
    object DataContext { get; set; }
}

public interface IViewModel
{
}

然后把他们都搞砸了

public sealed partial class MainWindow : Window, IView {...}
public partial class Page1 : Page, IView {...}
public partial class Page2 : Page, IView {...}

public class MainViewModel : NotifyPropertyChanged, IViewModel {...}
public class Page1ViewModel : NotifyPropertyChanged, IViewModel {...}
public class Page2ViewModel : NotifyPropertyChanged, IViewModel {...}

我用单例实现了这两个容器

public static class ViewService
{
    private static readonly ConcurrentDictionary<Type, IView> instances = new ConcurrentDictionary<Type, IView>();

    public static T GetView<T>(IViewModel viewModel = null) where T : IView
    {
        if (!instances.TryGetValue(typeof(T), out IView view))
        {
            view = (T)Activator.CreateInstance(typeof(T));
            instances[typeof(T)] = view;
        }
        if (viewModel != null)
            view.DataContext = viewModel;
        return (T)view;
    }
}

public static class ViewModelService
{
    private static readonly ConcurrentDictionary<Type, IViewModel> instances = new ConcurrentDictionary<Type, IViewModel>();

    public static T GetViewModel<T>() where T : IViewModel
    {
        if (!instances.TryGetValue(typeof(T), out IViewModel viewModel))
        {
            viewModel = (T)Activator.CreateInstance(typeof(T));
            instances[typeof(T)] = viewModel;
        }
        return (T)viewModel;
    }
}

我像这样创建主窗口

public partial class App : Application
{
    protected override void OnStartup(StartupEventArgs e)
    {
        base.OnStartup(e);
        MainViewModel vm = ViewModelService.GetViewModel<MainViewModel>();
        MainWindow window = ViewService.GetView<MainWindow>(vm);
        window.Show();
    }
}

并且页面在主视图模型的构造函数中

public class MainViewModel : NotifyPropertyChanged, IViewModel
{
    private IView _currentPage;

    public List<IView> Pages { get; }

    public IView CurrentPage
    {
        get => _currentPage;
        set
        {
            _currentPage = value;
            OnPropertyChanged();
            OnPropertyChanged(nameof(Title));
        }
    }

    public string Title => $"IoC Demo - {CurrentPage.Title}";

    public MainViewModel()
    {
        Pages = new List<IView>
        {
            ViewService.GetView<Page1>(ViewModelService.GetViewModel<Page1ViewModel>()),
            ViewService.GetView<Page2>(ViewModelService.GetViewModel<Page2ViewModel>())
        };
        CurrentPage = Pages[0];
    }
}

主窗口MainWindow

<Grid>
    <Grid.ColumnDefinitions>
        <ColumnDefinition Width="Auto"/>
        <ColumnDefinition/>
    </Grid.ColumnDefinitions>
    <ListBox BorderThickness="0" Background="AliceBlue" ItemsSource="{Binding Pages}" SelectedItem="{Binding CurrentPage}">
        <ListBox.ItemTemplate>
            <DataTemplate>
                <TextBlock Text="{Binding Title}" HorizontalAlignment="Center" VerticalAlignment="Center" Margin="5"/>
            </DataTemplate>
        </ListBox.ItemTemplate>
    </ListBox>
    <Frame Content="{Binding CurrentPage}" Grid.Column="1" NavigationUIVisibility="Hidden"/>
</Grid>

页面是一样的,只有一种类型Page1,第二种Page2

<Grid>
    <TextBlock Margin="5" Text="{Binding Text}"/>
</Grid>

页面视图模型(用于可见性)

public class Page1ViewModel : NotifyPropertyChanged, IViewModel
{
    public string Text { get; } = "Page one";
}

这一切都可以协同工作。

问题:我是否违反了 MVVM,我是否正确地使用单例实现了 IoC 容器?

国际奥委会 WPF

附言

  • WPF 中的 IoC 根据来自 @EvgeniyZ 和 @tym32167 的 MVVM 规则我已经阅读过。这很有帮助,谢谢!
c#
  • 1 1 个回答
  • 10 Views

1 个回答

  • Voted
  1. Best Answer
    aepot
    2020-10-02T05:26:21Z2020-10-02T05:26:21Z

    它似乎已经掌握了它。

    1. 构造函数发生了变化,以便他们采用特定类型的 View Model 并将其放入DataContext.
    public MainWindow(MainViewModel viewModel) {...}
    public Page1(Page1ViewModel viewModel) {...}
    public Page2(Page2ViewModel viewModel) {...}
    
    1. 接口IViewModel被丢弃为无用

    2. 实现了一个IoC容器,这次希望是真的

    public static class IocContainer
    {
        private static readonly IDictionary<Type, object> instances = new ConcurrentDictionary<Type, object>();
    
        public static void Register<T>() where T : class
            => instances[typeof(T)] = null;
    
        public static T Resolve<T>() where T : class
            => (T)GetInstance(typeof(T));
    
        private static object GetInstance(Type type)
        {
            bool registered = instances.TryGetValue(type, out object instance);
            if (instance == null)
            {
                object[] args = type.GetConstructors().First().GetParameters().Select(x => GetInstance(x.ParameterType)).ToArray();
                instance = Activator.CreateInstance(type, args);
                if (registered)
                    instances[type] = instance;
            }
            return instance;
        }
    }
    
    1. 代码OnStartup现在如下所示:
    protected override void OnStartup(StartupEventArgs e)
    {
        base.OnStartup(e);
        IocContainer.Register<MainViewModel>();
        IocContainer.Register<Page1ViewModel>();
        IocContainer.Register<Page2ViewModel>();
    
        IocContainer.Resolve<MainWindow>().Show();
    }
    
    1. 构造函数MainViewModel是这样的:
    public MainViewModel()
    {
        Pages = new List<IView>
        {
            IocContainer.Resolve<Page1>(),
            IocContainer.Resolve<Page2>()
        };
        CurrentPage = Pages[0];
    }
    

    感谢所有评论者,他们为我提供了非常必要的信息,尤其是SimpleIoC 提供了帮助。

    在 Yandex.Disk链接上使用此解决方案进行PS存档。

    • 3

相关问题

  • 使用嵌套类导出 xml 文件

  • 分层数据模板 [WPF]

  • 如何在 WPF 中为 ListView 手动创建列?

  • 在 2D 空间中,Collider 2D 挂在玩家身上,它对敌人的重量相同,我需要它这样当它们碰撞时,它们不会飞向不同的方向。统一

  • 如何在 c# 中使用 python 神经网络来创建语音合成?

  • 如何知道类中的方法是否属于接口?

Sidebar

Stats

  • 问题 10021
  • Answers 30001
  • 最佳答案 8000
  • 用户 6900
  • 常问
  • 回答
  • Marko Smith

    如何从列表中打印最大元素(str 类型)的长度?

    • 2 个回答
  • Marko Smith

    如何在 PyQT5 中清除 QFrame 的内容

    • 1 个回答
  • Marko Smith

    如何将具有特定字符的字符串拆分为两个不同的列表?

    • 2 个回答
  • Marko Smith

    导航栏活动元素

    • 1 个回答
  • Marko Smith

    是否可以将文本放入数组中?[关闭]

    • 1 个回答
  • Marko Smith

    如何一次用多个分隔符拆分字符串?

    • 1 个回答
  • Marko Smith

    如何通过 ClassPath 创建 InputStream?

    • 2 个回答
  • Marko Smith

    在一个查询中连接多个表

    • 1 个回答
  • Marko Smith

    对列表列表中的所有值求和

    • 3 个回答
  • Marko Smith

    如何对齐 string.Format 中的列?

    • 1 个回答
  • Martin Hope
    Alexandr_TT 2020年新年大赛! 2020-12-20 18:20:21 +0000 UTC
  • Martin Hope
    Alexandr_TT 圣诞树动画 2020-12-23 00:38:08 +0000 UTC
  • Martin Hope
    Air 究竟是什么标识了网站访问者? 2020-11-03 15:49:20 +0000 UTC
  • Martin Hope
    Qwertiy 号码显示 9223372036854775807 2020-07-11 18:16:49 +0000 UTC
  • Martin Hope
    user216109 如何为黑客设下陷阱,或充分击退攻击? 2020-05-10 02:22:52 +0000 UTC
  • Martin Hope
    Qwertiy 并变成3个无穷大 2020-11-06 07:15:57 +0000 UTC
  • Martin Hope
    koks_rs 什么是样板代码? 2020-10-27 15:43:19 +0000 UTC
  • Martin Hope
    Sirop4ik 向 git 提交发布的正确方法是什么? 2020-10-05 00:02:00 +0000 UTC
  • Martin Hope
    faoxis 为什么在这么多示例中函数都称为 foo? 2020-08-15 04:42:49 +0000 UTC
  • Martin Hope
    Pavel Mayorov 如何从事件或回调函数中返回值?或者至少等他们完成。 2020-08-11 16:49:28 +0000 UTC

热门标签

javascript python java php c# c++ html android jquery mysql

Explore

  • 主页
  • 问题
    • 热门问题
    • 最新问题
  • 标签
  • 帮助

Footer

RError.com

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

帮助

© 2023 RError.com All Rights Reserve   沪ICP备12040472号-5