深入解析 WinForms MVVM 模式中的事件驱动与数据驱动

前言

在传统的 WinForms 开发中,事件驱动模型(Event-Driven Model)是核心,它通过控件的事件(如点击按钮、改变文本等)触发业务逻辑。然而,MVVM 模式引入了数据驱动(Data-Driven)的理念,这种方式通过数据绑定实现视图和业务逻辑之间的解耦,使代码更清晰、更易于维护。本文将解析这两种驱动模型,并结合前面所述基于 MVVM 的 WinForms 框架示例来展示如何将两者结合起来,特别是增加数据驱动部分。
在这里插入图片描述

一、事件驱动模型

1. 什么是事件驱动模型?

事件驱动模型是一种基于用户交互或系统事件触发操作的编程模式。它依赖于事件的定义和监听,用户操作触发事件,然后由事件处理程序执行逻辑代码。

示例:登录按钮点击事件

public partial class LoginForm : Form
{
    private readonly LoginViewModel _viewModel;

    public LoginForm(LoginViewModel viewModel)
    {
        InitializeComponent();
        _viewModel = viewModel;
    }

    private void btnLogin_Click(object sender, EventArgs e)
    {
        _viewModel.Username = tb_user.Text.Trim();
        _viewModel.Password = tb_password.Text.Trim();
        _viewModel.LoginCommand.Execute(null);
        this.Hide();
    }
}

二、数据驱动模型

在这里插入图片描述

1. 什么是数据驱动模型?

数据驱动模型通过数据的变化来驱动 UI 的更新。通过数据绑定机制,视图和数据模型保持同步,无需显式的事件监听和处理。当数据发生变化时,UI 自动更新。

示例:数据驱动登录表单

在 MVVM 模式中,数据驱动的核心是通过数据绑定,使视图模型(ViewModel)中的数据改变时,视图(View)能够自动更新。以下是一个简单的例子来说明这一点。

1. ViewModel 示例

视图模型使用 INotifyPropertyChanged 接口,让属性改变时通知视图。

using System.ComponentModel;
using System.Runtime.CompilerServices;

public class LoginViewModel : INotifyPropertyChanged
{
    private string _username;
    private string _password;

    public event PropertyChangedEventHandler PropertyChanged;

    public string Username
    {
        get => _username;
        set
        {
            if (_username != value)
            {
                _username = value;
                OnPropertyChanged();
            }
        }
    }

    public string Password
    {
        get => _password;
        set
        {
            if (_password != value)
            {
                _password = value;
                OnPropertyChanged();
            }
        }
    }

    protected void OnPropertyChanged([CallerMemberName] string propertyName = null)
    {
        PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
    }
}
2. View 示例

视图绑定到 LoginViewModel,使用数据绑定让 UI 自动更新。

using System.Windows.Forms;

public partial class LoginForm : Form
{
    private readonly LoginViewModel _viewModel;

    public LoginForm(LoginViewModel viewModel)
    {
        InitializeComponent();
        _viewModel = viewModel;

        // 绑定数据到控件
        txtUsername.DataBindings.Add("Text", _viewModel, nameof(_viewModel.Username), false, DataSourceUpdateMode.OnPropertyChanged);
        txtPassword.DataBindings.Add("Text", _viewModel, nameof(_viewModel.Password), false, DataSourceUpdateMode.OnPropertyChanged);
    }
}

简要说明

  1. 当用户在 txtUsernametxtPassword 输入框中输入内容时,输入值会自动更新到 LoginViewModel 中的 UsernamePassword 属性。
  2. 通过数据绑定,视图中的控件和视图模型中的数据保持同步,简化了代码逻辑。

三、在 WinForms MVVM框架 中实现数据驱动

尽管 WinForms 没有内置的强大数据绑定机制(如 WPF),我们仍可以通过 INotifyPropertyChanged 接口实现数据驱动。以下是如何在示例框架中加入数据驱动功能。

1. 增加 INotifyPropertyChanged 实现

LoginViewModel 中实现 INotifyPropertyChanged 接口,让属性的改变能够通知绑定的 UI 控件。

using System.ComponentModel;
using System.Windows.Input;
using WinFormMVVM.Services;
using WinFormMVVM.Commands;
using WinFormMVVM.Views;
using System.Runtime.CompilerServices;
using System.Diagnostics;

namespace WinFormMVVM.ViewModels
{
    public class LoginViewModel : INotifyPropertyChanged
    {
        private string _username;
        private string _password;
        private bool _isPasswordInValid;
        private readonly IUserService _userService;

        public event PropertyChangedEventHandler PropertyChanged;

        public string Username
        {
            get => _username;
            set
            {
                if (_username != value)
                {
                    _username = value;
                    OnPropertyChanged();
                }
            }
        }
        public string Password
        {
            get => _password;
            set
            {
                if (_password != value)
                {
                    _password = value;
                    ValidatePassword(); // 验证密码
                    OnPropertyChanged();
                }
            }
        }

        public bool IsPasswordInValid
        {
            get => _isPasswordInValid;
            private set
            {
                if (_isPasswordInValid != value)
                {
                    _isPasswordInValid = value;
                    OnPropertyChanged();
                }
            }
        }
        private void ValidatePassword()
        {
            // 检查密码是否包含大写字母和小写字母
            IsPasswordInValid = string.IsNullOrEmpty(_password) ||(!(_password.Any(char.IsUpper) &&
                              _password.Any(char.IsLower)));
            Trace.WriteLine(IsPasswordInValid);
        }
        protected void OnPropertyChanged([CallerMemberName] string propertyName = null)
        {
            PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
        }

        public ICommand LoginCommand { get; }

        public LoginViewModel(IUserService userService)
        {
            _userService = userService;
            LoginCommand = new RelayCommand(Login);
        }

        private void Login()
        {
            var user = _userService.GetUserByUsername(Username);
            if (user != null && user.Password == Password)
            {
                MainForm mainForm = new MainForm();
                mainForm.Show();
            }
            else
            {
                MessageBox.Show("显示登录失败的消息!");
            }
        }
    }
}

2. 在视图中绑定数据

我们可以通过 DataBindings 来将控件属性绑定到 ViewModel

示例:绑定数据到文本框

using WinFormMVVM.ViewModels;

namespace WinFormMVVM.Views
{
    public partial class LoginForm : Form
    {
        private readonly LoginViewModel _viewModel;

        public LoginForm(LoginViewModel viewModel)
        {
            InitializeComponent();
            _viewModel = viewModel;
            tb_user.DataBindings.Add("Text", _viewModel, nameof(_viewModel.Username), false, DataSourceUpdateMode.OnPropertyChanged);
            tb_password.DataBindings.Add("Text", _viewModel, nameof(_viewModel.Password), false, DataSourceUpdateMode.OnPropertyChanged);
            lblPasswordValidation.DataBindings.Add("Visible", _viewModel, nameof(_viewModel.IsPasswordInValid), false, DataSourceUpdateMode.OnPropertyChanged);
            lblPasswordValidation.Text = "密码必须包含大小写字母";

        }

        private void btnLogin_Click(object sender, EventArgs e)
        {
            _viewModel.LoginCommand.Execute(null);
            this.Hide();
        }
    }
}

执行结果:
在这里插入图片描述

四、Winforms中支持数据绑定的控件

在 WinForms 中,许多控件都支持数据绑定,可以实现视图和数据的同步。以下是常用的支持数据绑定的控件:

1. TextBox

  • 用途: 用于显示和编辑文本。
  • 绑定属性: Text
txtUsername.DataBindings.Add("Text", viewModel, "Username", false, DataSourceUpdateMode.OnPropertyChanged);

2. Label

  • 用途: 用于显示只读文本。
  • 绑定属性: Text
lblMessage.DataBindings.Add("Text", viewModel, "StatusMessage");

3. CheckBox

  • 用途: 用于表示布尔值(选中或未选中)。
  • 绑定属性: Checked
chkRememberMe.DataBindings.Add("Checked", viewModel, "RememberMe");

4. ComboBox

  • 用途: 显示下拉列表。
  • 绑定属性: SelectedValue, SelectedItem, Text
cmbOptions.DataBindings.Add("SelectedValue", viewModel, "SelectedOption");
cmbOptions.DataSource = viewModel.Options; // 绑定数据源

5. ListBox

  • 用途: 显示列表项。
  • 绑定属性: SelectedItem, SelectedValue
lstItems.DataBindings.Add("SelectedItem", viewModel, "SelectedItem");
lstItems.DataSource = viewModel.Items; // 绑定数据源

6. DataGridView

  • 用途: 显示表格数据。
  • 绑定属性: DataSource
dataGridView.DataBindings.Add("DataSource", viewModel, "TableData");

7. PictureBox

  • 用途: 显示图片。
  • 绑定属性: Image
pictureBox.DataBindings.Add("Image", viewModel, "ProfilePicture");

8. ProgressBar

  • 用途: 显示任务进度。
  • 绑定属性: Value
progressBar.DataBindings.Add("Value", viewModel, "Progress");

9. DateTimePicker

  • 用途: 显示和选择日期和时间。
  • 绑定属性: Value
dateTimePicker.DataBindings.Add("Value", viewModel, "SelectedDate");

10. TrackBar

  • 用途: 显示和选择数值范围中的值。
  • 绑定属性: Value
trackBar.DataBindings.Add("Value", viewModel, "VolumeLevel");

WinForms 中的大多数控件都支持通过 DataBindings 来实现数据驱动的功能。通过使用 DataBindings.Add 方法,可以将视图模型中的属性与控件的属性绑定,从而在视图模型的数据发生变化时自动更新视图。

五、事件驱动与数据驱动结合的优势

1. 清晰的职责划分

  • 事件驱动:专注于用户交互和事件处理。
  • 数据驱动:专注于数据变化和 UI 同步。

2. 降低代码耦合

  • 视图和逻辑解耦,视图模型仅关心业务逻辑,视图关心展示。

3. 更易于测试

  • 视图模型独立于 UI,易于编写单元测试。

六、总结

在 WinForms 中实现 MVVM 模式,可以结合事件驱动和数据驱动模型,发挥各自的优势。通过 INotifyPropertyChanged 和数据绑定,我们可以实现更加简洁、可维护的 WinForms 应用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/911318.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Python中的 if __name__ == ‘__main__

在Python中,if __name__ __main__: 这行代码有一个特定的用途和含义。为了理解它,我们需要先了解Python中的__name__变量以及Python脚本是如何执行的。 __name__变量 每个Python模块(python文件)都有一个内置的属性__name__。当…

【系统面试篇】进程与线程类(2)(笔记)——进程调度、中断、异常、用户态、核心态

目录 一、相关面试题 1. 进程的调度算法有哪些? 调度原则 (1)先来先服务调度算法 (2)最短作业优先调度算法 (3)高响应比优先调度算法 (4)时间片轮转调度算法 &am…

大数据工具 flume 的安装配置与使用 (详细版)

参考网址:Flume 1.9用户手册中文版 — 可能是目前翻译最完整的版本了 1,上传安装包 安装包链接:文件下载-奶牛快传 Download |CowTransfer 口令:x8bhcg 1,切换盘符到安装目录 cd /opt/moudles 解压文件…

使用uni-app框架开发各种web前端程序

使用uni-app框架开发各种web前端程序是目前非常流程的开发方式,比如开发APP、小程序、H5等等,是一个使用 vue 开发所有前端应用的框架,开发者编写一套代码,可发布到ios,安卓、H5、以及各种小程序(微信、支付…

鸿蒙ArkTS中的布局容器组件(Column、Row、Flex、 Stack、Grid)

在鸿蒙ArkTS中,布局容器组件有很多,常见的有:   ⑴ Column:(垂直布局容器):用于将子组件垂直排列。   ⑵ Row:(水平布局容器):用于将子组件水…

简单介绍一下mvvm mvc mvp以及区别、历史

MVC(Model - View - Controller) 因MVC架构的灵活性,架构图形式很多,仅供参考 历史: MVC 是最早出现的软件架构模式之一,其历史可以追溯到 20 世纪 70 年代,最初被用于 Smalltalk - 80 环境。…

黑马程序员linux学习【持续更新】

Linux基础 一、Linux简介 1.分类 不同领域的主流操作系统,主要分为下 几类:桌面操作系统、服务器操作系统、移动设备操作系统、嵌入式操作系统。 桌面操作系统 操作系统特点Windows用户数量最多MacOS操作体验好,办公人士首选Linux用户数…

Vert.x,应用监控 - 全链路跟踪,基于Zipkin

关于Zipkin Zipkin是一款开源的分布式实时数据追踪系统(Distributed Tracking System),能够收集服务间调用的时序数据,提供调用链路的追踪。Zipkin每一个调用链路通过一个trace id来串联起来,通过trace id,就能够直接定位到这次调…

ENSP作业——园区网

题目 根据上图,可得需求为: 1.配置交换机上的VLAN及IP地址。 2.设置SW1为VLAN 2/3的主根桥,设置SW2为VLAN 20/30的主根桥,且两台交换机互为主备。 3.可以使用super vlan。 4.上层通过静态路由协议完成数据通信过程。 5.AR1作为企…

从壹开始解读Yolov11【源码研读系列】——Data.dataset.py:模型训练数据预处理/YOLO官方数据集类——YOLODataset

【前情回顾】在上一篇文章记录了YOLO源码data目录下的 base.py 文件,其中定义了一个可灵活修改的数据加载处理基类——Class BaseDataset 灵活基类博文地址:https://blog.csdn.net/qq_58718853/article/details/143249295 【实验代码】所有实验代码上传至…

HFSS 3D Layout中Design setting各个选项的解释

从HFSS 3D LAYOUT菜单中,选择Design Settings打开窗口,会有六个选项:DC Extrapolation, Nexxim Options, Export S Parameters, Lossy Dielectrics, HFSS Meshing Method, and HFSS Adaptive Mesh. DC Extrapolation 直流外推 直流外推分为标…

【板栗糖GIS】——如果安装的vscode版本落后了,如何无障碍更新

【板栗糖GIS】——如果安装的vscode版本落后了,如何无障碍更新 今天想安装新扩展插件时发现vscode版本有点旧,于是在不影响插件的情况下更新。 打开vscode软件,点击设置检查更新,如果有需要更新的直接安装到原目录,如…

数字化转型必看!华为数字化最全合集(192页PDF限免下载 )

今天给大家整理了6份关于华为数字化的资料,共计192页,干货满满! 资料已经全部打包,划到文末添加大师兄即可免费下载👇👇👇 一、华为实施数字化转型方法论与实践的业务解读 这份报告是华为实施数…

【数据集】【YOLO】【目标检测】道路结冰数据集 1527 张,YOLO目标检测实战训练教程!

数据集介绍 【数据集】道路结冰数据集 1527 张,目标检测,包含YOLO/VOC格式标注。数据集中包含2种分类:“clear_road, ice_road”。数据集来自国内外图片网站和视频截图,部分数据经过数据增强处理。检测范围监控视角检测、无人机视…

创建线程时传递参数给线程

在C中,可以使用 std::thread 来创建和管理线程,同时可以通过几种方式将参数传递给线程函数。这些方法包括使用值传递、引用传递和指针传递。下面将对这些方法进行详细讲解并给出相应的代码示例。 1. 值传递参数 当你创建线程并希望传递参数时&#xff…

集智书童 | DuoDiff: 提升浅层 Transformer 性能的扩散模型, 双 Backbone 件扩散模型在图像处理中的应用 !

本文来源公众号“集智书童”,仅用于学术分享,侵权删,干货满满。 原文链接:DuoDiff: 提升浅层 Transformer 性能的扩散模型, 双 Backbone 件扩散模型在图像处理中的应用 ! 扩散模型在图像生成方面取得了前所…

linux perf 环境部署和基本测试(基于Ubuntu20.04)

1,linux 安装perf sudo apt-ge install linux-tools-common sudo apt-get install linux-tools-$(uname -r) linux-tools-generic -y 2 补充安装 sudo apt-get install python3-q-text-as-data 3,perf常用命令 larkubuntu:~$ perf usage: perf [--version] [--hel…

PHP露营地管理平台小程序系统源码

⛺️【露营新风尚】露营地管理平台系统全攻略⛺️ 🏕️一、露营热潮下的管理难题:如何高效运营露营地?🤔 随着露营文化的兴起,越来越多的人选择在大自然中享受宁静与自由。然而,露营地的管理却面临着诸多…

信息安全工程师(83)Windows操作系统安全分析与防护

一、Windows操作系统安全分析 系统漏洞: Windows操作系统由于其复杂性和广泛使用,可能存在一些已知或未知的漏洞。这些漏洞可能会被黑客利用,进行恶意攻击。微软会定期发布系统更新和补丁,以修复这些漏洞,提高系统的安…

软件测试—功能测试详解

🍅 点击文末小卡片 ,免费获取软件测试全套资料,资料在手,涨薪更快 一、测试项目启动与研读需求文档 (一) 组建测试团队 1、测试团队中的角色 2、测试团队的基本责任 尽早地发现软件程序、系统或产…