WPF系列之一:基于并行任务和MVVM创建响应灵敏和数据驱动的UI
在利用WPF创建桌面应用程序的界面时,经常使用MVVM的设计模式,以减少UI层与逻辑层的代码耦合度。
在MVVM的设计中,最主要的方法和技术是UI中的控件利用Binding来和逻辑层(ViewModel)进行交互,其中控件的属性为依赖属性,而作为控件的DataContext的ViewModel则实现了INotifyPropertyChanged接口。
除了一般意义上的属性的数据的交互,还有一些基于命令的Banding来实现界面元素的操作与内部函数的解耦。
这样,界面上的数据和操作(包括控件的命令和事件属性,事件的解耦后面会有一篇文章说明,在这里)都可以和底层实现解耦了,这样就使得ViewModel层和UI层有了明显的界限,松耦合的实现对将来的功能扩展和单元测试会是非常大的便利。
此外,因为View层(控件层)与ViewModel层的解耦,使得原来利用控件的事件进行底层交互的操作(可以使用Dispatcher进行异步交互)全部移到ViewModel层中了。某些情形下,这种交互可能是非常费时间的,比如访问数据库或者进行密集型运算,此时就可以利用.net的并行库对Model层进行异步的调用,当Model层结束查询或者运算时将结果更新到ViewModel层,ViewModel层因为实现了INotifyPropertyChanged接口,使得UI层得到通知更新。体现了数据驱动界面的思想。
准备:下载 Prism ,利用其中的DelegateCommand 放到ViewModel层来做为View层控件的Command绑定目标。
下面给出实际的例子:
1.创建一个WPF程序:
2.从下载的Prism包中找到Bin文件夹下的Desktop目录,引用其中的Assembly Microsoft.Practices.Prism.dll到新建的工程
3.主窗口对应的xaml文件如下:
<Window x:Class="WPF.MVVMDemo.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:vm="clr-namespace:WPF.MVVMDemo"
Title="MVVMDemo" Height="600" Width="600">
<Window.DataContext>
<vm:MainViewModel></vm:MainViewModel>
</Window.DataContext>
<Grid>
<StackPanel>
<Button Height="30" Command="{Binding CalculateCommand}">Start to run complex calculation in engine</Button>
<Button Command="{Binding CalculateCommandWithParameter}" CommandParameter="I am the parameter command!" Content="Start to run complex calculation in engine with Parameter" Height="30" />
<TextBlock Height="30" Text="{Binding CalculatingStatus}"></TextBlock>
<GroupBox Header="Data from engine" Height="200" Name="groupBox1" Width="Auto">
<Grid>
<TextBox Height="220" TextWrapping="Wrap" Text="{Binding DataStringFromEngine}"></TextBox>
</Grid>
</GroupBox>
<GroupBox Header="Always editable Box" Height="200" Name="groupBox2" Width="Auto">
<Grid>
<TextBox Height="150" TextWrapping="Wrap" Text="You can edit me while calculating"></TextBox>
</Grid>
</GroupBox>
</StackPanel>
</Grid>
</Window>
MainWindow.xaml
4.对应窗口的分部代码不用改动。
5.定义一个类做为窗口的ViewModel
public class MainViewModel : INotifyPropertyChanged, IDisposable
{
public event PropertyChangedEventHandler PropertyChanged;
private ModelSimulator engine;
public MainViewModel()
{
dataStringFromEngine = string.Empty;
calculatingStatus = string.Empty;
isEngineFree = true;
engine = new ModelSimulator();
calculateCommand = new DelegateCommand(this.executeCalculateCommand, this.canExecuteCalculateCommand);
calculateCommandWithParameter = new DelegateCommand<string>(this.executeCalculateCommandWithParameter, this.canExecuteCalculateCommandWithParameter);
cancelCalculateCommand = new DelegateCommand(this.executeCancelCalculateCommand, this.canExecuteCancelCalculateCommand);
}
private string dataStringFromEngine;
public string DataStringFromEngine
{
get { return dataStringFromEngine; }
set
{
if (dataStringFromEngine != value)
{
dataStringFromEngine = value;
OnPropertyChagned("DataStringFromEngine");
}
}
}
private string calculatingStatus;
public string CalculatingStatus
{
get { return calculatingStatus; }
set
{
if (calculatingStatus != value)
{
calculatingStatus = value;
OnPropertyChagned("CalculatingStatus");
}
}
}
private bool isEngineFree;
public bool IsEngineFree
{
get { return isEngineFree; }
set
{
if (isEngineFree != value)
{
isEngineFree = value;
OnPropertyChagned("IsEngineFree");
((DelegateCommand)calculateCommand).RaiseCanExecuteChanged();
((DelegateCommand<string>)calculateCommandWithParameter).RaiseCanExecuteChanged();
((DelegateCommand)cancelCalculateCommand).RaiseCanExecuteChanged();
CommandManager.InvalidateRequerySuggested();
}
}
}
private ICommand calculateCommand;
public ICommand CalculateCommand
{
get { return calculateCommand; }
set
{
if (calculateCommand != value)
{
calculateCommand = value;
OnPropertyChagned("CalculateCommand");
}
}
}
private void executeCalculateCommand()
{
IsEngineFree = false;
CalculatingStatus = "Running!";
// create parallel task ,async Task<string> engineTask = Task.Factory.StartNew<string>(() => engine.SimulateLongTimeWork());
//UI callback
engineTask.ContinueWith(task =>
{ this.DataStringFromEngine = task.Result;
IsEngineFree = true;
CalculatingStatus = "Complete!";
});
}
private bool canExecuteCalculateCommand()
{
return isEngineFree;
}
private ICommand calculateCommandWithParameter;
public ICommand CalculateCommandWithParameter
{
get { return calculateCommandWithParameter; }
set
{
if (calculateCommandWithParameter != value)
{
calculateCommandWithParameter = value;
OnPropertyChagned("CalculateCommandWithParameter");
}
}
}
private void executeCalculateCommandWithParameter(string para)
{
IsEngineFree = false;
CalculatingStatus = "Running!";
// create parallel task ,async
Task<string> engineTask = Task.Factory.StartNew<string>(() => engine.SimulateLongTimeWorkWithParameter(para, ));
//UI callback
engineTask.ContinueWith(task =>
{ this.DataStringFromEngine = task.Result;
IsEngineFree = true;
CalculatingStatus = "Complete!";
}); }
private bool canExecuteCalculateCommandWithParameter(string para)
{
return isEngineFree;
}
private ICommand cancelCalculateCommand;
public ICommand CancelCalculateCommand
{
get { return cancelCalculateCommand; }
set
{
if (cancelCalculateCommand != value)
{
cancelCalculateCommand = value;
OnPropertyChagned("CancelCalculateCommand");
}
}
}
private void executeCancelCalculateCommand()
{
}
private bool canExecuteCancelCalculateCommand()
{
return !isEngineFree;
}
private void OnPropertyChagned(string propertyName)
{
if (PropertyChanged != null)
{
this.PropertyChanged.Invoke(this, new PropertyChangedEventArgs(propertyName));
}
}
public void Dispose()
{
//throw new NotImplementedException();
}
}
MainViewModel
6.定义一个模拟Model层的类ModelSimulator
public class ModelSimulator
{
public string SimulateLongTimeWork(int seconds)
{
string rtn = null;
Random rnd = new Random(DateTime.Now.Millisecond);
for (int i = ; i < ; i++)
{ rtn = rtn + rnd.Next(-, );
}
Thread.Sleep(new TimeSpan(,,seconds));
return rtn;
}
public string SimulateLongTimeWorkWithParameter(string para, int seconds)
{
string rtn = para + Environment.NewLine;
Random rnd = new Random(DateTime.Now.Millisecond);
for (int i = ; i < ; i++)
{
rtn = rtn + rnd.Next(-, );
}
Thread.Sleep(new TimeSpan(, , seconds));
return rtn;
}
}
ModelSimulator
最主要的代码就是创建带返回值的异步任务与Model层进行交互,命令被调用时,界面会保持响应状态。而Model层交互完成之后更新ViewModel层的数据。
此模式称为Future模式(带返回值得异步调用)。
// create parallel task ,async
Task<string> engineTask = Task.Factory.StartNew<string>(() => engine.SimulateLongTimeWorkWithParameter(para, ));
//UI callback
engineTask.ContinueWith(task =>
{ this.DataStringFromEngine = task.Result;
IsEngineFree = true;
CalculatingStatus = "Complete!";
});
作者:Andy Zeng
欢迎任何形式的转载,但请务必注明出处。
http://www.cnblogs.com/andyzeng/p/3701892.html
WPF系列之一:基于并行任务和MVVM创建响应灵敏和数据驱动的UI的更多相关文章
- WPF系列教程——(二)使用Prism实现MVVM设计模式 - 简书
原文:WPF系列教程--(二)使用Prism实现MVVM设计模式 - 简书 本文假设你已经知道MVVM设计模式是什么,所以直接进入正题,今天我们就用Prism来实现WPF的MVVM设计模式,百度上关于 ...
- 在WPF中使用依赖注入的方式创建视图
在WPF中使用依赖注入的方式创建视图 0x00 问题的产生 互联网时代桌面开发真是越来越少了,很多应用都转到了浏览器端和移动智能终端,相应的软件开发上的新技术应用到桌面开发的文章也很少.我之前主要做W ...
- [WPF系列]-数据邦定之DataTemplate 根据对象属性切换模板
引言 书接上回[WPF系列-数据邦定之DataTemplate],本篇介绍如何根据属性切换模板(DataTemplate) 切换模板的两种方式: 使用DataTemplateSelecto ...
- [WPF系列]从基础起步学习系列计划
引言 WPF技术已经算不什么新技术,一搜一大把关于WPF基础甚至高级的内容.之前工作中一直使用winform所以一直没有深入学习WPF,这次因项目中使用了WPF技术来实现比较酷的展示界面.我在这里只是 ...
- 《Programming WPF》翻译 第8章 5.创建动画过程
原文:<Programming WPF>翻译 第8章 5.创建动画过程 所有在这章使用xaml举例说明的技术,都可以在代码中使用,正如你希望的.可是,代码可以使用动画在某种程度上不可能在x ...
- 《Programming WPF》翻译 第6章 1.创建和使用资源
原文:<Programming WPF>翻译 第6章 1.创建和使用资源 资源这个词具有非常广泛的意义.任何对象都可以是一个资源.一个在用户界面中经常使用的Brush或者Color可以是一 ...
- JavaScript系列-----对象基于哈希存储(<Key,Value>之Value篇) (3)
JavaScript系列-----Objectj基于哈希存储<Key,Value>之Value 1.问题提出 在JavaScript系列-----Object之基于Hash<Key, ...
- 基于xlua和mvvm的unity框架
1.框架简介 这两天在Github上发现了xlua的作者车雄生前辈开源的一个框架—XUUI,于是下载下来学习了一下.XUUI基于xlua,又借鉴了mvvm的设计概念.xlua是目前很火的unity热更 ...
- saltstack自动化运维系列11基于etcd的saltstack的自动化扩容
saltstack自动化运维系列11基于etcd的saltstack的自动化扩容 自动化运维-基于etcd加saltstack的自动化扩容# tar -xf etcd-v2.2.1-linux-amd ...
随机推荐
- 剑指 Offer——和为 S 的两个数字
1. 题目 2. 解答 由于数组是已经排好序的,我们可以定义两个指针,第一个指针指向第一个元素,第二个指针指向最后一个元素,然后求出这两个元素的和,与目标和进行比较.若小于目标和,第一个指针向前移动: ...
- NO.3:自学python之路------集合、文件操作、函数
引言 本来计划每周完成一篇Python的自学博客,由于上一篇到这一篇遇到了过年.开学等杂事,导致托更到现在.现在又是一个新的学期,春天也越来越近了(冷到感冒).好了,闲话就说这么多.开始本周的自学Py ...
- 【转】node.js框架比较
我偶然间看到这篇文章,转这个文章并没有什么含义,仅仅是感觉总结的不错,对于新学node的友友们来说希望这篇文章为大家对 Node.js 后端框架选型带来一些帮助,学习不再迷茫,也是让我有个保存,以后参 ...
- HDU 5433 Xiao Ming climbing 动态规划
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5433 Xiao Ming climbing Time Limit: 2000/1000 MS (Ja ...
- lintcode-202-线段树的查询
202-线段树的查询 对于一个有n个数的整数数组,在对应的线段树中, 根节点所代表的区间为0-n-1, 每个节点有一个额外的属性max,值为该节点所代表的数组区间start到end内的最大值. 为Se ...
- linux桌面使用鼠标中间健粘帖
使用linux桌面很久了,一直习惯鼠标左键选中,右健弹出菜单复制粘帖. 没想到linux使用鼠标中间健粘帖,很方便. 参考:Linux鼠标中键复制粘贴之谜[Felix蛋疼科普贴] 用鼠标左键单击待复制 ...
- PAT---福尔摩斯约会时间
主要为字符串的处理,注意读懂题目意思. 设置输出域宽和填充字符的函数分别为setw(int n),setfill(char c);两个函数的头文件为#include<iomanip>; # ...
- React-native APK打包
安卓相关工具配置到环境变量,这样可以将安卓相关工具可以直接在cmd命令中调用 1 检查gradle版本 查看里面对应的编译工具版本号,如果提示版本不对你,那么直接去更新android sdk,相关的s ...
- 【第二周】关于java.util包下的Random类
1.功能:此类的实例用于生成伪随机数流 2.方法(Random的方法有很多,在此只解释说明我认为比较常用的几个方法) (1)next(int bits):生成下一个伪随机数 (2)nextDouble ...
- jenkin重新注册用户
http://www.cnblogs.com/xiao-fy/