百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT知识 > 正文

界面控件DevExpress WinForm - MVVM服务讲解(二)

liuian 2025-03-05 17:58 7 浏览

DevExpress Universal Subscription官方最新版免费下载试用,历史版本下载,在线文档和帮助文件下载-慧都网

考虑像显示来自 ViewModel 的通知(例如,消息框)这样的微不足道的任务,作为可视化元素,任何消息框实际上都是视图的一部分。 因此,如果你直接从 ViewModel 显示消息框(定义一个调用 MessageBox.Show() 方法的命令),这个简单的代码将破坏主要的MVVM概念 - ViewModels不能引用Views,并使其无法编写ViewModel的单元测试。为了解决这个困难,DevExpress MVVM 框架实现了服务。

服务是一种 IOC 模式,它删除了 ViewModel 和 View 层之间的任何引用。 在代码中,服务是在 ViewModel 代码中使用的接口,没有任何关于“何时”和“如何”实现该接口的假设。

您可以实现自己的自定义服务以及使用 DevExpress Services,无论您使用什么服务,通用工作流程都保持不变:

  • 在代码中定义服务(如果您使用的是 DevExpress 已经实现的服务,则跳过);
  • 在特定的视图中注册它;
  • 在ViewModel中检索服务并使用其方法。

自定义服务

对于自定义服务,您首先需要在单独类中的某处实现此服务,例如应用程序具有带有 Notify 方法的自定义接口 IMyNotificationService。

C#

//View
public interface IMyNotificationService {
void Notify(string message);
}

VB.NET

'View
Public Interface IMyNotificationService
Sub Notify(ByVal message As String)
End Interface

然后,实现此接口的自定义服务 CustomService1 将如下所示。

C#

//Service1.cs
class CustomService1 : IMyNotificationService {
void IMyNotificationService.Notify(string message) {
System.Windows.Forms.MessageBox.Show(message, "Service1");
}
}

VB.NET

'Service1.vb
Friend Class CustomService1
Implements IMyNotificationService

Private Sub IMyNotificationService_Notify(ByVal message As String) Implements IMyNotificationService.Notify
System.Windows.Forms.MessageBox.Show(message, "Service1")
End Sub
End Class

作为变体,创建另一个实现相同接口但使用不同方法重载的服务。

C#

//Service2.cs
class CustomService2 : IMyNotificationService {
void IMyNotificationService.Notify(string message) {
System.Windows.Forms.MessageBox.Show(message, "Service2");
}
}

VB.NET

'Service2.vb
Friend Class CustomService2
Implements IMyNotificationService

Private Sub IMyNotificationService_Notify(ByVal message As String) Implements IMyNotificationService.Notify
System.Windows.Forms.MessageBox.Show(message, "Service2")
End Sub
End Class

在 ViewModel 代码中检索自定义服务的属性将如下所示。

C#

//ViewModel
public virtual IMyNotificationService Service {
get { throw new NotImplementedException(); }
}

public virtual IMyNotificationService AnotherService {
get { throw new NotImplementedException(); }
}

VB.NET

'ViewModel
Public Overridable ReadOnly Property Service() As IMyNotificationService
Get
Throw New NotImplementedException()
End Get
End Property

Public Overridable ReadOnly Property AnotherService() As IMyNotificationService
Get
Throw New NotImplementedException()
End Get
End Property

这是可以绑定到 UI 元素(例如,按钮)的 DoSomething 方法,它将显示具有相同文本的两条消息。

C#

//ViewModel
public void DoSomething() {
Service.Notify("Hello");
AnotherService.Notify("Hello");
}

VB.NET

'ViewModel
Public Sub DoSomething()
Service.Notify("Hello")
AnotherService.Notify("Hello")
End Sub

最后,在视图中注册您的自定义服务。 由于这些是您自己的自定义服务,因此不存在用于注册这些服务的预定义静态 MVVMContext 方法。 相反,调用本地 MvvmContext 实例的 RegisterService 方法。

C#

//View
mvvmContext1.RegisterService(new CustomService1());
mvvmContext1.RegisterService(new CustomService2());

VB.NET

'View
mvvmContext1.RegisterService(New CustomService1())
mvvmContext1.RegisterService(New CustomService2())

提示:注册后,服务在分层树中占据特定位置。每当框架需要服务时,它就会从树的底部开始寻找,向上移动直到找到合适的服务。前面有提到很多现成的服务已经在静态容器中注册了,这些服务位于层次结构的最顶层,如果框架没有在较低级别找到任何自定义服务,则会使用这些服务。如果这两个默认服务都不存在,则会发生异常。在此示例中,两个自定义服务都注册在同一层次结构级别上。由于这两个服务实现了相同的 IMyNotificationService 服务,因此在调用 Service 或 AnotherService 对象的 Notify 方法时,它们都被视为合适的服务。但是 CustomService2 是最后注册的,因此它更靠近层次结构底部,并且总是首先被框架“找到”。您可以欺骗此机制并使用 RegisterDefaultService 方法注册 CustomService2,这将在层次结构顶部的静态容器中注册您的 CustomService2,并使 CustomSerive1 成为最底层的服务。之后,框架将始终选择 CustomService1。

为了解决这个问题,您可以定义服务密钥。key是标记特定服务的字符串标识符,对于 POCO ViewModel,您可以将服务密钥设置为 [ServiceProperty] 属性的参数。

C#

[ServiceProperty(Key="Service1")]
public virtual IMyNotificationService Service {
get { throw new NotImplementedException(); }
}

[ServiceProperty(Key = "Service2")]
public virtual IMyNotificationService AnotherService {
get { throw new NotImplementedException(); }
}

VB.NET


Public Overridable ReadOnly Property Service() As IMyNotificationService
Get
Throw New NotImplementedException()
End Get
End Property


Public Overridable ReadOnly Property AnotherService() As IMyNotificationService
Get
Throw New NotImplementedException()
End Get
End Property

对于非 POCO ViewModel,可以将服务密钥设置为 GetService 扩展方法的参数。

C#

public IServiceInterface MyService {
get { return this.GetService("MyServiceKey"); }
}

VB.NET

Public ReadOnly Property MyService() As IServiceInterface
Get
Return Me.GetService(Of IServiceInterface )("MyServiceKey")
End Get
End Property

现在,您必须使用这些唯一密钥注册您的自定义服务,所有 Register 方法都有相应的重载来做到这一点。

C#

mvvmContext1.RegisterService("Service1", new CustomService1());
mvvmContext1.RegisterService("Service2", new CustomService2());

VB.NET

mvvmContext1.RegisterService("Service1", New CustomService1())
mvvmContext1.RegisterService("Service2", New CustomService2())

因此,当您调用 Notify 方法时,框架将不会混淆应该使用哪个 IMyNotificationService 服务实现。 相反它将采用标有您的自定义密钥的确切服务,例如,AnotherService 属性将寻找标有“Service2”键的服务,并找到已注册的 CustomService2 服务。 与 Service 属性相同,它将使用 CustomService1 服务,因为它标有“Service1”键。

如果您现在测试应用程序,您将看到两个消息框现在显示不同的标题,因为它们是由不同服务的方法显示的。

DevExpress WinForm

DevExpress WinForm拥有180+组件和UI库,能为Windows Forms平台创建具有影响力的业务解决方案。DevExpress WinForms能完美构建流畅、美观且易于使用的应用程序,无论是Office风格的界面,还是分析处理大批量的业务数据,它都能轻松胜任!



相关推荐

【常识】如何优化Windows 7

优化Windows7可以让这个经典系统运行更流畅,特别是在老旧硬件上。以下是经过整理的实用优化方案,分为基础优化和进阶优化两部分:一、基础优化(适合所有用户)1.关闭不必要的视觉效果右键计算机...

系统优化!Windows 11/10 必做的十个优化配置

以下是为Windows10/11用户整理的10个必做优化配置,涵盖性能提升、隐私保护和系统精简等方面,操作安全且无需第三方工具:1.禁用不必要的开机启动项操作路径:`Ctrl+S...

最好用音频剪辑的软件,使用方法?

QVE音频剪辑是一款简单实用的软件,功能丰富,可编辑全格式音频。支持音频转换、合并、淡入淡出、变速、音量调节等,无时长限制,用户可自由剪辑。剪辑后文件音质无损,支持多格式转换,便于存储与跨设备播放,满...

Vue2 开发总踩坑?这 8 个实战技巧让代码秒变丝滑

前端开发的小伙伴们,在和Vue2打交道的日子里,是不是总被各种奇奇怪怪的问题搞得头大?数据不响应、组件传值混乱、页面加载慢……别慌!今天带来8个超实用的Vue2实战技巧,每一个都能直击痛...

Motion for Vue:为Vue量身定制的强大动画库

在前端开发中,动画效果是提升用户体验的重要手段。Vue生态系统中虽然有许多动画库,但真正能做到高性能、易用且功能丰富的并不多。今天,我们要介绍的是MotionforVue(motion-v),...

CSS view():JavaScript 滚动动画的终结

前言CSSview()方法可能会标志着JavaScript在制作滚动动画方面的衰落。如何用5行CSS代码取代50多行繁琐的JavaScript,彻底改变网页动画每次和UI/U...

「大数据」 hive入门

前言最近会介入数据中台项目,所以会推出一系列的跟大数据相关的组件博客与文档。Hive这个大数据组件自从Hadoop诞生之日起,便作为Hadoop生态体系(HDFS、MR/YARN、HIVE、HBASE...

青铜时代的终结:对奖牌架构的反思

作者|AdamBellemare译者|王强策划|Tina要点运维和分析用例无法可靠地访问相关、完整和可信赖的数据。需要一种新的数据处理方法。虽然多跳架构已经存在了几十年,并且可以对...

解析IBM SQL-on-Hadoop的优化思路

对于BigSQL的优化,您需要注意以下六个方面:1.平衡的物理设计在进行集群的物理设计需要考虑数据节点的配置要一致,避免某个数据节点性能短板而影响整体性能。而对于管理节点,它虽然不保存业务数据,但作...

交易型数据湖 - Apache Iceberg、Apache Hudi和Delta Lake的比较

图片由作者提供简介构建数据湖最重要的决定之一是选择数据的存储格式,因为它可以大大影响系统的性能、可用性和兼容性。通过仔细考虑数据存储的格式,我们可以增强数据湖的功能和性能。有几种不同的选择,每一种都有...

深入解析全新 AWS S3 Tables:重塑数据湖仓架构

在AWSre:Invent2024大会中,AWS发布了AmazonS3Tables:一项专为可扩展存储和管理结构化数据而设计的解决方案,基于ApacheIceberg开放表格...

Apache DataFusion查询引擎简介

简介DataFusion是一个查询引擎,其本身不具备存储数据的能力。正因为不依赖底层存储的格式,使其成为了一个灵活可扩展的查询引擎。它原生支持了查询CSV,Parquet,Avro,Json等存储格式...

大数据Hadoop之——Flink Table API 和 SQL(单机Kafka)

一、TableAPI和FlinkSQL是什么TableAPI和SQL集成在同一套API中。这套API的核心概念是Table,用作查询的输入和输出,这套API都是批处理和...

比较前 3 名Schema管理工具

关注留言点赞,带你了解最流行的软件开发知识与最新科技行业趋势。在本文中,读者将了解三种顶级schema管理工具,如AWSGlue、ConfluentSchemaRegistry和Memph...

大数据技术之Flume

第1章概述1.1Flume定义Flume是Cloudera提供的一个高可用的,高可靠的,分布式的海量日志采集、聚合和传输的系统。Flume基于流式架构,灵活简单。1.2Flume的优点1.可以和...