WPF真入门教程23--MVVM简单介绍

这篇具有很好参考价值的文章主要介绍了WPF真入门教程23--MVVM简单介绍。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

       

  在WPF开发中,经典的编程模式是MVVM,是为WPF量身定做的模式,该模式充分利用了WPF的数据绑定机制,最大限度地降低了Xmal文件和CS文件的耦合度,也就是UI显示和逻辑代码的耦合度,如需要更换界面时,逻辑代码修改很少,甚至不用修改。与WinForm开发相比,我们一般在后置代码中会使用控件的名字来操作控件的属性来更新UI,而在WPF中通常是通过数据绑定来更新UI;在响应用户操作上,WinForm是通过控件的事件来处理,而WPF可以使用命令绑定的方式来处理,耦合度将降低,WPF技术的主要特点是数据驱动UI,所以在使用WPF技术开发的过程中是以数据为核心的,WPF提供了数据绑定机制,当数据发生变化时,WPF会自动发出通知去更新UI。本教程的第15到19简单介绍了数据绑定的简单应用

一、MVVM介绍

  MVVM是Model-View-ViewModel(模型-视图-视图模型)的缩写形式,它通常被用于WPF或Silverlight开发。我们可以通过下图来直观的理解MVVM模式:

wpf进阶之mvvm教程,WPF真入门教程,wpf,ui

1、View就是xaml界面,也就是那个扩展名为xaml的文件,它负责与用户交互,接收用户输入,把数据展现给用户。

2、ViewModel就是一个C#类,负责收集需要绑定的数据和命令,通过View类的DataContext属性绑定到View,同时也可以处理一些UI逻辑。

3、Model,就是数据库系统中的实体对象,可包含属性和行为。

  三者之间的关系:View对应一个ViewModel,ViewModel可以聚合N个Model,ViewModel可以对应多个View

二、MVVM的优势

      MVVM的根本思想就是界面和业务功能进行分离,View的职责就是负责如何显示数据及发送命令,ViewModel的功能就是如何提供数据和执行命令。各司其职,互不影响。在实际的业务场景中我们经常会遇到客户对界面提出建议要求修改,使用MVVM模式开发,当设计的界面不满足客户时,我们仅仅只需要对View作修改,不会影响到ViewModel中的功能代码,减少了犯错的机会。随着功能地增加,系统越来越复杂,相应地程序中会增加View和ViewModel文件,将复杂的界面分离成局部的View,局部的View对应局部的ViewModel,功能点散落在各个ViewModel中,每个ViewModel只专注自己职能之内的事情。ViewModel包含了View要显示的数据,并且知道View的交互代码,使用MVVM架构具有以下优势:

1、易维护 2、灵活扩展  3、易测试  4、用户界面设计师与程序开发者能更好的合作

三、MVVM简单示例

1、在项目的model文件创建类Emp及EmpViewModel,内容如下:

wpf进阶之mvvm教程,WPF真入门教程,wpf,ui

wpf进阶之mvvm教程,WPF真入门教程,wpf,ui

注意这个EmpViewModel叫“视图模型”,通过数据绑定将它们绑在一起,它真的是一个很好的适配器能将模型变成某种WPF框架可以使用的东西。所以这个就是MVVM中的VM,而Emp是M,同在缺少V。

2、在control文件夹中添加窗口MVVMWindow1.xaml,具体内容如下:

<Window x:Class="WpfApp6.control.MVVMWindow1"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
        xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
        xmlns:local="clr-namespace:WpfApp6.control"
        xmlns:vm="clr-namespace:WpfApp6.model"
        mc:Ignorable="d"
        Title="MVVM简单介绍" Height="450" Width="800">
    <!--窗口的上下文声明一个EmpViewModel的实例,前面必须定义vm这个命名空间-->
    <Window.DataContext>
        <vm:EmpViewModel></vm:EmpViewModel>
    </Window.DataContext>
    <Window.Resources>
        <!--声明样式-->
        <Style x:Key="st1">
            <Setter Property="Control.FontSize" Value="17"></Setter>
        </Style>
    </Window.Resources>
    <Grid  Style="{StaticResource st1}" >
        <Button  Content="更新" HorizontalAlignment="Left" Margin="180,256,0,0" VerticalAlignment="Top" Width="145" Click="Button_Click"/>
        <Label Content="用户:" HorizontalAlignment="Left" Margin="112,110,0,0" VerticalAlignment="Top" RenderTransformOrigin="7.256,0.582"/>
        <!--Binding绑定视图模型中的UserName和CompanyName-->
        <TextBox  HorizontalAlignment="Left" Height="23" Margin="205,110,0,0" TextWrapping="Wrap" Text="{Binding UserName}" VerticalAlignment="Top" Width="120"/>
        <TextBox HorizontalAlignment="Left" Height="23" Margin="555,110,0,0" TextWrapping="Wrap" Text="{Binding CompanyName}" VerticalAlignment="Top" Width="120"/>
        <Label Content="公司:" HorizontalAlignment="Left" Margin="393,110,0,0" VerticalAlignment="Top"/>

    </Grid>
</Window>

 3、MVVMWindow1.xaml.cs后台代码:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Shapes;
using WpfApp6.model;

namespace WpfApp6.control
{
    /// <summary>
    /// MVVMWindow1.xaml 的交互逻辑
    /// </summary>
    public partial class MVVMWindow1 : Window
    {
        EmpViewModel empvm ;
        public MVVMWindow1()
        {
            InitializeComponent();
           empvm = base.DataContext as EmpViewModel;
        }
        /// <summary>
        /// 单击按钮
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void Button_Click(object sender, RoutedEventArgs e)
        {
            empvm.UserName = "欧阳克";
            empvm.CompanyName = "华山庄";
        }
    }
}

 4、启动程序,记得将项目启动文件App.xaml中的启动项改为 StartupUri="control/MVVMWindow1.xaml",运行结果,点击按钮时,不会有任何反应

wpf进阶之mvvm教程,WPF真入门教程,wpf,ui

 这里我们点击更新按钮不会有任何反应,因为还没有实现数据绑定。此时视图不会收到任何的关于属性改变的通知。要解决这个问题我们必须实现名称为INotifyPropertyChanged的接口。任何实现了这个接口的类,当属性发生改变的时候会通知所有监听者,所以我们需要修改视图模型NameViewModel类,完整代码是:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace WpfApp6.model
{
    /// <summary>
    /// 页面视图模型,实现INotifyPropertyChanged接口
    /// </summary>
    public class EmpViewModel:INotifyPropertyChanged
    {
        //构造方法
        public EmpViewModel()
        {
            emp = new Emp() { UserName = "杨康", CompanyName = "上海金财" };
        }
        //私有字段 
        private Emp emp;
        //公共属性
        public string UserName
        {
            get { return emp.UserName; }
            set { emp.UserName = value;RaisePropertyChanged("UserName"); }
        }
        // 属性值发生更改时触发的属性
        public event PropertyChangedEventHandler PropertyChanged; 
        //属性更改方法
        private void RaisePropertyChanged(string propertyName)
        { 
            if (PropertyChanged != null)
            {
                PropertyChanged.Invoke(this, new PropertyChangedEventArgs(propertyName));
            }
        }

        public string CompanyName
        {
            get { return emp.CompanyName; }
            set { emp.CompanyName = value; RaisePropertyChanged("CompanyName"); }
        }
    }
}

注意视图模型类实现了INotifyPropertyChanged接口,即通知属性更改,每个属性中增加了代码RaisePropertyChanged,这个方法就是属性更改时的方法,并且定义了一个事件属性,即public event PropertyChangedEventHandler PropertyChanged;其他代码没有变,运行后,点击更新效果是:wpf进阶之mvvm教程,WPF真入门教程,wpf,ui

 

wpf进阶之mvvm教程,WPF真入门教程,wpf,ui

 这里说明UI同步更新了,关键在于INotifyPropertyChanged,但是上面的方法还有不足,那就是假如有多个model需要设置属性更改,那工作量是蛮麻烦和巨大的,这里需要改进方法,定义一个基类,让基类实现INotifyPropertyChanged,让用户类继续基类,具体操作是:

5、在model中添加类ViewModelBase,代码是:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Linq;
using System.Runtime.CompilerServices;
using System.Text;
using System.Threading.Tasks;

namespace WpfApp6.model
{
    /// <summary>
    /// 模型视图其类
    /// </summary>
    public class ViewModelBase : INotifyPropertyChanged
    {
        /// <summary>
        /// 属性值发生更改时触发
        /// </summary>
        public event PropertyChangedEventHandler PropertyChanged;
        /// <summary>
        /// 执行更改
        /// C#5.0中的新特性CallerMemberName
        /// </summary>
        /// <param name="propertyName"></param>
        protected void OnPropertyChanged([CallerMemberName] string propertyName = "")
        {
            if (PropertyChanged != null)
            {
                PropertyChanged.Invoke(this, new PropertyChangedEventArgs(propertyName));
            }
        }
    }
}

6、在model创建另一个类,EmpViewModel2,代码 是:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace WpfApp6.model
{
    /// <summary>
    /// 继承基类ViewModelBase
    /// </summary>
    public class EmpViewModel2:ViewModelBase
    {
        public EmpViewModel2()
        {
            emp = new Emp() { UserName = "杨康", CompanyName = "上海金财" };
        }
        //私有字段 
        private Emp emp;
        //公共属性
        public string UserName
        {
            get { return emp.UserName; }
            set { 
                emp.UserName = value; 
                OnPropertyChanged(); 
            }
        }
        public string CompanyName
        {
            get { return emp.CompanyName; }
            set { 
                emp.CompanyName = value; 
                OnPropertyChanged(); 
            }
        }
    }
}

 6、修改MVVMWindow1.xaml文件的一个地方,即改成以下内容,表示现在窗口的上下文使用EmpViewModel2,也就是上面第5步的视图模型,而不是原来的EmpViewModel,其他都不要改

  wpf进阶之mvvm教程,WPF真入门教程,wpf,ui

 7、修改MVVMWindow1.xaml.cs代码,如图

 wpf进阶之mvvm教程,WPF真入门教程,wpf,ui

 8、运行程序,效果如下:

wpf进阶之mvvm教程,WPF真入门教程,wpf,ui

 小结,我们在实际项目中应该使用后面定义基类的方式去实现属性的绑定,这才是对的。

操作起来6666,棒棒哒。文章来源地址https://www.toymoban.com/news/detail-774174.html

到了这里,关于WPF真入门教程23--MVVM简单介绍的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • WPF 入门教程DockPanel介绍

    在 DockPanel中 可以很容易地停靠在所有四个方向的内容(上,下,左,右)。这使它在许多情况下成为一个很好的选择,您希望将窗口划分为特定区域,特别是因为默认情况下,DockPanel 内的最后一个元素,除非此功能被明确禁用,否则将自动填充其余空间(中心)。 我们在

    2024年02月05日
    浏览(31)
  • WPF真入门教程01--WPF简介

            Windows Presentation Foundation (简称 WPF),WPF是微软推出的基于Windows 的用户界面框架,属于.NET Framework 3.0的一部分。它提供了统一的编程模型、语言和框架,真正做到了分离界面设计人员与开发人员的工作;同时它提供了全新的多媒体交互用户图形界面。因与“我佩服”拼

    2024年02月06日
    浏览(41)
  • WPF入门到精通:3.MVVM简单应用及全局异常处理

    在WPF应用程序开发中,MVVM(Model-View-ViewModel)是一种非常流行的架构模式。它为应用程序的设计提供了良好的分层结构和可扩展性。 结构分为下列三部分 Model:定义了应用程序的数据模型 就是系统中的对象,可包含属性和行为(是一个class实体,是对现实中事物的抽象,开

    2024年02月11日
    浏览(26)
  • WPF真入门教程02--新建WPF工程

    在VS开发环境安装完成之后,首先我们先新建一个WPF工程,然后对工程目录结构啥的要有所了解才行。 打开VS2019      工程建好之后,WPF应用程序”会在“引用”里面自动添加下图中所示的 PresentationCore、PresentationFramework、WindowsBase三大核心程序集,就是下面这个样子   默认

    2024年02月03日
    浏览(51)
  • MaterialDesignInXAML WPF入门教程 快速入门

    先去MaterialDesignInXAML下载下来源码,以及Releases,在DemoApp 中就可以看到实际的效果很惊艳了。 除了要有一定的C#、winform 基础外,建议先学习一下 XAML,对整个开发环境有个基础的了解,再来学习此教程。 可以去bilibili上免费学习一下。教程一共12个小时,如果不看后面的实战

    2024年02月05日
    浏览(39)
  • WPF入门教程系列一——基础

    一、 前言            最近在学习WPF,学习WPF首先上的是微软的MSDN,然后再搜索了一下网络有关WPF的学习资料。为了温故而知新把学习过程记录下来,以备后查。这篇主要讲WPF的开发基础,介绍了如何使用Visual Studio 2013创建一个WPF应用程序。 首先说一下学习WPF的基础知

    2024年02月07日
    浏览(38)
  • WPF教程_编程入门自学教程_菜鸟教程-免费教程分享

    WPF教程 WPF - 概述 WPF - 环境设置 WPF - Hello World WPF - XAML概述 WPF - Elements Tree WPF - 依赖属性 WPF - 路由事件 WPF - 控件 WPF - 布局 WPF - 布局嵌套 WPF - 输入 WPF - 命令行 WPF - 数据绑定 WPF - 资源 WPF - 模板 WPF - 样式 WPF - 触发器 WPF - 调试 WPF - 自定义控件 WPF - 异常处理 WPF - 本地化 WPF - 互

    2023年04月27日
    浏览(37)
  • WPF 入门教程DispatcherTimer计时器

    https://www.zhihu.com/tardis/bd/art/430630047?source_id=1001 在 WinForms 中,有一个名为 Timer 的控件,它可以在给定的时间间隔内重复执行一个操作。WPF 也有这种可能性,但我们有 DispatcherTimer 控件,而不是不可见的控件。它几乎做同样的事情,但不是将它放在表单上,​​而是专门从代码

    2024年01月22日
    浏览(33)
  • WPF 入门教程Grid使用技巧

    在上一章中,我们向您介绍了出色的 Grid 面板,并向您展示了一些有关如何使用它的基本示例。在本章中,我们将进行一些更高级的布局,因为这是 Grid 真正闪耀的地方。首先,让我们加入更多的列甚至一些行,以获得真正的表格布局: 总共九个按钮,每个按钮都放置在自己

    2024年02月06日
    浏览(34)
  • WPF真入门教程12--ListView控件

           ListView 控件在Windows应用程序中常用,用于表示数据列表。如果您以前使用过 WinForms,那么您对ListView的实用性有一个很好的了解,但您应该意识到 WPF中的ListView 不像WinForms版本那样使用。再一次的主要区别在于,虽然WinForms ListView只是调用Windows API 函数来呈现常见的

    2024年02月04日
    浏览(36)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包