RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / 问题 / 1586750
Accepted
Pavel
Pavel
Asked:2024-07-10 21:49:06 +0000 UTC2024-07-10 21:49:06 +0000 UTC 2024-07-10 21:49:06 +0000 UTC

从 XAML 端连接到命令的最佳方式是什么?

  • 772

我通常像这样进行绑定 - Command="{Binding SomeCommand}",其中 SomeCommand 是来自 DataContext 的命令。

但在这个教育项目https://github.com/xellans/LearnMvvm.git中,我遇到了一个对我来说不寻常的选项

<RadioButton Command="{local:SetCurrentContext}" CommandParameter="PersonVM" Content="{DynamicResource Lang1}"></RadioButton>

这是绑定命令的可接受的方式吗?哪种方法被认为更正确?

c#
  • 1 1 个回答
  • 46 Views

1 个回答

  • Voted
  1. Best Answer
    EvgeniyZ
    2024-07-11T00:50:20Z2024-07-11T00:50:20Z

    为了使此绑定起作用,您的命令(以及要绑定的任何其他对象)必须是静态的,打开代码,查看,确实:

    public static readonly RoutedUICommand SetCurrentContext = new RoutedUICommand(
        "Задание текущего контекста навигатору, находящемуся в ресурсах, вызвашего команду, элемента.",
        "SetCurrentContext",
        typeof(NavigatorLocator));
    

    嗯,静态是邪恶的,因为至少

    • 静力学测试不佳
    • 即使我们根本不需要这个对象,静态也会永远挂在内存中
    • 嗯等等,需要考虑的微妙之处

    我并不是说你不应该有静态,我是说如果有这样的机会,最好不要有静态,在这种特殊情况下,有这样的机会。

    现在考虑一下如何实现其他命令来绑定视图{local:SetCurrentContext}?产生大量静态对象?我想你自己也明白这是一个愚蠢的决定。

    因此,对于单次使用,如定位器的示例(我们稍后会讨论),这种方法可能是有意义的,但是将所有属性设置为静态绑定就是搬起石头砸自己的脚。如果您更详细地研究示例中的代码,您将看到更多的普通命令和普通绑定,例如,AuthUserControl.xaml它包含Command="{Binding AuthorizeCommand}",并且在 VM 中已经有一个标准的非静态属性public RelayCommand AuthorizeCommand { get; }。

    现在,至于定位器
    ,老实说,我不喜欢这种方法,原因如下:

    1. 静态 - 正如我上面所写,我反对它,定位器通常基于它。
    2. 定位器通常嵌入在资源中(视图层),由 XAML 管理它们(创建、存储、管理生命周期等),这意味着我们失去了对它们的完全控制,这是非常糟糕的。这也会产生如所示示例中的拐杖(文件App.xaml.cs-> (NavigatorLocator)this.FindResource("locator");)。
    3. 为了一些可以更简单地完成的事情,代码变得非常复杂。

    当您有一个非常大的项目时,定位器是很好的选择,但这不是事实,因为如上面的示例所示,该项目使用 IoC 容器,通过它我们可以轻松请求我们需要的一切。因此,我个人不会使用这样的定位器,但是否值得由您决定。

    现在我将向您展示为什么我认为“定位器”是多余的。
    让我们以我们的项目将使用 IoC 容器为基础(在建议的示例中,这是InstancesProvider,在我的例子中它将是库)Microsoft.Extensions.DependencyInjection。

    • 创建一个新项目

    • 打开App.xaml和收起StartupUri="MainWindow.xaml"

    • 让我们为我们的窗口创建一个类,让它成为标准的MainViewModel

    • 我们打开App.xaml.cs并重新定义那里OnStartup,在其中指定容器的初始化,从中获取窗口和VM,然后打开窗口,我们得到如下内容:

      protected override void OnStartup(StartupEventArgs e)
      {
          base.OnStartup(e);
      
          var serviceProvider = new ServiceCollection()
              .AddTransient<MainViewModel>()
              .AddTransient<MainWindow>()
              .BuildServiceProvider();
      
          var window = serviceProvider.GetService<MainWindow>();
          var viewModel = serviceProvider.GetService<MainViewModel>();
      
          window!.DataContext = viewModel;
          window!.Show();
      }
      

      这是一个“笨拙”的实现,有充分的理由应该有一个单独的服务负责“映射”,我们在这里询问并通过它打开,但为了举例(且不仅仅是),我认为它可以。

    • 让我们运行该项目并看到一个空窗口。恭喜,我们已经创建了一个带有 IoC 容器的项目。

    现在让我们考虑内容(页面),让一个单独的服务负责它们,这将通知侦听器页面已更改。

    • 我们创建一个接口,作为VM页面层的“联系人”,让它简单而空interface IPage。

    • 现在让我们创建另一个接口来描述服务本身,或者更确切地说什么是“公共”,即更改内容的方法,甚至是一个事件,如下所示:

      public interface IContentService
      {
          event Action<IPage>? OnPageChanged;
          void Change<TPage>() where TPage : IPage;
      }
      
    • 现在让我们尽可能简单地实现服务本身,如下所示:

      public class ContentService (IServiceProvider serviceProvider) : IContentService
      {
          public event Action<IPage>? OnPageChanged;
      
          public void Change<TPage>() where TPage : IPage
          {
              var page = serviceProvider.GetService<TPage>();
              if (page is null) return;
              OnPageChanged?.Invoke(page);
          }
      }
      
    • 在容器中注册服务.AddSingleton<IContentService, ContentService>()

    • 现在我们来实现MainViewModel

      public partial class MainViewModel : ObservableObject
      {
          public MainViewModel(IContentService contentService) 
              => contentService.OnPageChanged += page => CurrentPage = page;
      
          [ObservableProperty]
          private IPage? _currentPage;
      }
      

      不用管ObservableObject这个ObservableProperty,你可以在这里进行任何实现INotifyPropertyChanged。就我而言,它被使用CommunityToolkit.MVVM。

    • 让我们用 XAML 编写窗口<ContentPresenter Content="{Binding CurrentPage}" />;这对于示例来说就足够了。

    现在我想你可以看到我们的代码是多么简单;我们创建了一个没有任何资源、静态对象的服务,并实现了它,现在我们可以在任何地方使用它。现在看一下该定位器并回答“为什么?”的问题......我个人无法回答)

    我认为编写进一步的示例是没有意义的(如果您需要它,请告诉我,我会添加它),但一般来说,我们只需为必要的页面创建一个虚拟机,然后继承它们IPage并将每个页面注册到容器中,并且IContentService如果我们需要更改页面,也可以在所需的 VM 中(不仅如此)实现它,因此我们调用Change()具有所需 VM 类型的页面的方法来打开 ( contentService.Change<LoginVM>();)。

    实际上,这些是我对此事的想法,我想我已经清楚地解释了为什么我不喜欢这样的绑定,以及为什么我不太喜欢定位器,尤其是当我手头有一个 IoC 容器时......

    • 2

相关问题

  • 使用嵌套类导出 xml 文件

  • 分层数据模板 [WPF]

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

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

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

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

Sidebar

Stats

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

    我看不懂措辞

    • 1 个回答
  • Marko Smith

    请求的模块“del”不提供名为“default”的导出

    • 3 个回答
  • Marko Smith

    "!+tab" 在 HTML 的 vs 代码中不起作用

    • 5 个回答
  • Marko Smith

    我正在尝试解决“猜词”的问题。Python

    • 2 个回答
  • Marko Smith

    可以使用哪些命令将当前指针移动到指定的提交而不更改工作目录中的文件?

    • 1 个回答
  • Marko Smith

    Python解析野莓

    • 1 个回答
  • Marko Smith

    问题:“警告:检查最新版本的 pip 时出错。”

    • 2 个回答
  • Marko Smith

    帮助编写一个用值填充变量的循环。解决这个问题

    • 2 个回答
  • Marko Smith

    尽管依赖数组为空,但在渲染上调用了 2 次 useEffect

    • 2 个回答
  • Marko Smith

    数据不通过 Telegram.WebApp.sendData 发送

    • 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