Windows Phone 十一、MVVM模式
模型-视图-视图模型 (MVVM) 是一种用来分离 UI 和非 UI 代码的应用设计模式
MVVM 中的 Model 与 MVC 中的一致,用于封装业务逻辑以及数据处理。
Model 不依赖 View 和 ViewModel,可以独立存在,也就是说模型不关心自身被谁操作和展示。
Model 中不允许有任何跟界面相关的逻辑,比如操作界面上的控件。
通常在实际开发过程中 Model 可以再被划分成 Model 和 Service,区分业务和数据。
视图仅仅负责界面展示。
通过 DataContext(数据上下文)进行数据绑定。
不允许直接与 Model 交互。
可以通过绑定 Comand 来调用 ViewModel 的行为。
Command 是 View 到 ViewModel 的单向通行。
用户在 View 中触发事件,在 ViewModel 中处理。
ViewModel 是对 View 的抽象
视图模型包括界面模型数据和 Command 事件响应。
是 View 和 Model 的桥梁,是对 Model 的包装和抽象。
实现视图模型需要让类型实现 INotifyPropertyChanged 接口,用于实现属性和集合的变更通知,使用户在 View 上所做的操作可以实时通知到视图模型。
利用 MVVM 实现一个计算器;
要求界面和业务逻辑完全剥离;
比如我们可以更改界面展示形式,功能仍然相同;
第一步:先对页面进行抽象(建模)
第二步:让视图模型实现INotifyPropertyChanged接口
第三步:后台代码中声明一个视图模型属性交给当前数据上下文
第四步:前台数据绑定
第五步:定义Command命令
<Page
x:Class="MyJiSuanQi.MainPage"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:MyJiSuanQi"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="d"
DataContext="{Binding ViewModel,RelativeSource={RelativeSource Mode=Self}}"
Background="{ThemeResource ApplicationPageBackgroundThemeBrush}"> <StackPanel>
<Slider
Header="参数1"
Value="{Binding Parameter1,Mode=TwoWay}"/>
<!--<TextBox
x:Name="txtP1"
Header="参数1"
Text="{Binding Parameter1,Mode=TwoWay}"/>-->
<TextBox
x:Name="txtP2"
Header="参数2"
Text="{Binding Parameter2,Mode=TwoWay,UpdateSourceTrigger=PropertyChanged}"/>
<AppBarButton
x:Name="btnAdd"
Icon="Italic"
Label="Except"
Command="{Binding ExceptCommand}"
CommandParameter="{Binding}"/>
<TextBox
x:Name="txtResult"
Header="结果"
Text="{Binding Result,Mode=OneWay}"/>
</StackPanel>
</Page>
namespace MyJiSuanQi
{
/// <summary>
/// 可用于自身或导航至 Frame 内部的空白页。
/// </summary>
public sealed partial class MainPage : Page
{
public MainPageViewModel ViewModel { get; set; }
public MainPage()
{
ViewModel = new MainPageViewModel();
this.InitializeComponent(); this.NavigationCacheMode = NavigationCacheMode.Required;
} /// <summary>
/// 在此页将要在 Frame 中显示时进行调用。
/// </summary>
/// <param name="e">描述如何访问此页的事件数据。
/// 此参数通常用于配置页。</param>
protected override void OnNavigatedTo(NavigationEventArgs e)
{
// TODO: 准备此处显示的页面。 // TODO: 如果您的应用程序包含多个页面,请确保
// 通过注册以下事件来处理硬件“后退”按钮:
// Windows.Phone.UI.Input.HardwareButtons.BackPressed 事件。
// 如果使用由某些模板提供的 NavigationHelper,
// 则系统会为您处理该事件。
}
//第一步:先对页面进行抽象(建模)
//第二步:让视图模型实现INotifyPropertyChanged接口
//第三步:后台代码中声明一个视图模型属性交给当前数据上下文
//第四步:前台数据绑定
//第五步:定义Command命令
private void btnAdd_Click(object sender, RoutedEventArgs e)
{
//var d1 = double.Parse(txtP1.Text);
//var d2 = double.Parse(txtP2.Text);
//txtResult.Text = (d1 + d2).ToString();
ViewModel.Result = ViewModel.Parameter1 + ViewModel.Parameter2;
}
}
public class MainPageViewModel : INotifyPropertyChanged
{
public MainPageViewModel()
{
AddCommand = new AddCommand();
ExceptCommand = new ExceptCommand();
}
public double Parameter1 { get; set; }
private double parameter2;
//影响当前命令是否可以被执行的依据
public double Parameter2
{
get { return parameter2; }
set
{
if (parameter2 == value)
{
return;
}
parameter2 = value;
//通知命令对象重新判断是否可以执行
ExceptCommand.OnCanExecuteChanged();
}
}
//public double Result { get; set; }
private double result;
public double Result
{
get { return result; }
set { result = value; OnPropertyChanged("Result"); }
}
public ICommand AddCommand { get; set; }
public ExceptCommand ExceptCommand { get; set; }
private void OnPropertyChanged(string propertyName)
{
if (PropertyChanged != null)
{
PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
}
}
public event PropertyChangedEventHandler PropertyChanged;
}
//命令对象必须实现ICommand接口
public class AddCommand : ICommand
{
//判断当前命令对象是否可以被执行
public bool CanExecute(object parameter)
{
throw new NotImplementedException();
} public event EventHandler CanExecuteChanged;
//如何执行
public void Execute(object parameter)
{
var model = parameter as MainPageViewModel;
model.Result = model.Parameter1 + model.Parameter2;
}
}
//命令对象必须实现ICommand接口
public class ExceptCommand : ICommand
{
//判断当前命令对象是否可以被执行
public bool CanExecute(object parameter)
{
var model = parameter as MainPageViewModel;
return model != null && model.Parameter2 != ;
}
//当CanExecute判断依据发生变化时,触发事件
public event EventHandler CanExecuteChanged;
public void OnCanExecuteChanged()
{
if (CanExecuteChanged != null)
CanExecuteChanged(this, EventArgs.Empty);
}
//如何执行
public void Execute(object parameter)
{
var model = parameter as MainPageViewModel;
model.Result = model.Parameter1 / model.Parameter2;
}
}
}
低耦合:View 可以独立于 Model 变化和修改,一个 ViewModel 可以绑定到不同的 View 上,当 View 变化的时候 Model 可以不变,当 Model 变化的时候 View 也可以不变。
可重用性:可以把一些视图的逻辑放在 ViewModel 里面,让很多View重用这段视图逻辑。
独立开发:开发人员可以专注与业务逻辑和数据的开发(ViewModel)。设计人员可以专注于界面(View)的设计。
可测试性:可以针对ViewModel来对界面(View)进行测试。
Windows Phone 十一、MVVM模式的更多相关文章
- WPF自学入门(十一)WPF MVVM模式Command命令 WPF自学入门(十)WPF MVVM简单介绍
WPF自学入门(十一)WPF MVVM模式Command命令 在WPF自学入门(十)WPF MVVM简单介绍中的示例似乎运行起来没有什么问题,也可以进行更新.但是这并不是我们使用MVVM的正确方式 ...
- 重新想象 Windows 8 Store Apps (55) - 绑定: MVVM 模式
[源码下载] 重新想象 Windows 8 Store Apps (55) - 绑定: MVVM 模式 作者:webabcd 介绍重新想象 Windows 8 Store Apps 之 绑定 通过 M ...
- 企业级架构 MVVM 模式指南 (WPF 和 Silverlight 实现) 译(2)
本书包含的章节内容 第一章:表现模式,以一个例子呈献给读者表现模式的发展历程,我们会用包括MVC和MVP在内的各种方式实现一个收费项目的例子.沿此方向,我们会发现每一种模式的问题所在,这也是触发设计模 ...
- MVVM模式解析和在WPF中的实现(三)命令绑定
MVVM模式解析和在WPF中的实现(三) 命令绑定 系列目录: MVVM模式解析和在WPF中的实现(一)MVVM模式简介 MVVM模式解析和在WPF中的实现(二)数据绑定 MVVM模式解析和在WPF中 ...
- [转载]MVVM模式原理分析及实践
没有找到很好的MVVM模式介绍文章,简单找了一篇,分享一下.MVVM实现了UI\UE设计师(Expression Blend 4设计界面)和软件工程师的合理分工,在SilverLight.WPF.Wi ...
- silverlighter下MVVM模式中利用Behavior和TargetedTriggerAction实现文本框的一些特效
在silverlight一般开发模式中,给文本框添加一些事件是轻而易举的,然而MVVM开发模式中,想要给文本框添加一些事件并非那么容易,因为MVVM模式中,只有ICommand接口,而且也只有Butt ...
- 在 WPF 程序中使用 MVVM 模式
MVVM 模式是一个很久之前的技术了,最近因为一个项目的原因,需要使用 WPF 技术,所以,重新翻出来从前的一段程序,重温一下当年的技术. MVVM 模式 MVVM 实际上涉及三个部分,Model, ...
- MVVM模式应用 之的RelayCommand的使用
实现MVVM模式Command是立下了汗马功劳.当然ICommand要引用using System.Windows.Input命名空间. 比如: (1)我们在xaml页面有一个Button按钮,我们需 ...
- 什么是MVVM模式
问题引入1 场景一:团队辛辛苦苦完成了一个项目,抱着激动的心情去给用户做demo,而用户给你的反馈是UI很不满意,要重新修改,否则拒绝验收.大规模修改UI,晴天霹雳!2 场景二:产品在一家客户上线运行 ...
随机推荐
- JavaScript中的this指向
this是谁 技术一般水平有限,有什么错的地方,望大家指正. this代指当前对象super调用父类的构造函数,应表会运网数物,加载驱动建立链接执行SQL处理结果,直到现在每想起这三点就能想起我上大学 ...
- 你应该在开始API开发之前知道的事(上)(翻译)
这篇文章的源地址:http://dev.dota2.com/showthread.php?t=58317 由于文章内容较多,英语水平有限,准备尝试着以中英混搭的形式翻译,免得曲解一些不懂内容的意思.以 ...
- 【BZOJ】3521: [Poi2014]Salad Bar
题意 长度为\(n(1 \le n \le 1000000)\)的\(01\)字符串.找一个最长的连续子串\(S\),使得不管是从左往右还是从右往左取,都保证每时每刻已取出的\(1\)的个数不小于\( ...
- JavaScript之闭包就是个子公司
在计算机科学中,闭包(Closure)是词法闭包(Lexical Closure)的简称,是引用了自由变量的函数.这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外.所以,有 ...
- linux 关于session缓存丢失,自己掉坑里面了
突然间session失效了,死活打不上缓存.顺着解决思路:程序--再检查程序--文件权限--查看服务器配置 我只想说一句,我操.尼玛,各种调试,各种自虐.毫无意义,三个钟头流失. 看看几点了,快凌晨两 ...
- 重学ps_1
1,选取 打开图片->点击选取工具->ctrl+c->ctrl+n->ctrl+v 2,去除图片背景 打开你要去除背景的图片->在图层面板中->双击图层把图层改为0 ...
- 第八周PSP
团队项目PSP 一:表格 C类型 C内容 S开始时间 E结束时间 I时间间隔 T净时间(mins) 预计花费时间(mins) 讨论 讨论用户界面 8:40 11:40 35 60 70 分析与 ...
- Redmi Note3 hennessy 刷机过程记录
本文只是凭记忆,记录大致的步骤,提供线索. 准备 刷机包和supersu刷机包,到xiaomi.eu上下载, 如果是稳定版可能有锁bootloader,需要到下载解锁软件.开发版无锁 刷入recove ...
- Educational Codeforces Round 6 E dfs序+线段树
题意:给出一颗有根树的构造和一开始每个点的颜色 有两种操作 1 : 给定点的子树群体涂色 2 : 求给定点的子树中有多少种颜色 比较容易想到dfs序+线段树去做 dfs序是很久以前看的bilibili ...
- backbonejs
前言: backbone由以下模块组成. 一.Event 监听事件,自定义事件.绑定到任何对象. http://www.css88.com/doc/backbone/#Events 这个是下面模块核心 ...