MVP的PV模式与SC模式
MVC是现今挺被推崇的一种架构模式,而MVP在MVC的基础对视图与模型上再解耦,使结构和各自的功能也有所改变。在去年鄙人也尝试了一下使用MVP做了一个小Demo,作为了MVC的一个变体,MVP还分了两种模式,这个之前是不了解的,直到看了蒋老师的著作才知道。
在MVP里面重点还是看View与Presenter之间的交互,View可以直接去调用Presenter,但Presenter对View的调用不是直接去访问View的实例,而是通过由View去实现的一个IView接口。唉MVP中能看到的就是这种对象与对象之间的交互,在细化一点去看,处理UI的逻辑部分搁在View里面还是搁在Presenter里面。这就是PV(Passive View被动视图)模式和(Supervising Controller监督控制器)模式的区别了。那么先来对两种模式稍作介绍,然后再通过一个Demo再体验一下两种模式。
PV模式
在PV模式中,视图上的所有View的数据(具体点来说可以是控件的属性)都会暴露出来,供Presenter来调用。View的任务就是简单地在需要的时候把数据呈现出来,或者把数据提供给Presenter。涉及到一些UI逻辑的,无论复杂与否都与View本身无关。UI逻辑是Presenter的任务,这种方式使得View以一种很轻量级的形式存在,UI逻辑放到了Presenter里面,这样的做法有一种好处,就是方便了对UI逻辑做测试。但不足之处往往在于为了让Presenter对View数据操作更粒度更细化,View会提供比较多的属性,这样会使得IView接口变得庞大起来。
SC模式
在SC模式中鄙人感觉最为受到强调的就是数据流动的单向性。无论是Presenter把数据推送到View这个方向,还是View接收到界面操作然后提交到Presenter,都应该是要单向的。也就是说当Prensenter把消息推送到View之后,View不要再给Presenter反馈信息(返回值),View把界面操作的响应提交到Presenter之后也不需要Presenter回传点什么。
那么下面就看看实际的例子
这是一个简单的通讯录,点击左边联系人名单,在右边就会显示出联系人的姓名和电话
对于View的布局而言,两者都是一样的
![]() |
![]() |
| 设计界面 | 运行效果 |
还有另一个相同的就是模型Model,这里的模型只有两个属性,一个是姓名,另一个是电话号码,还提供了一个静态方法去创建一个联系人的列表,代码定义如下
public class ContactModel
{
public string PersionName { get; set; } public string PhoneNumber { get; set; } public static List<ContactModel> GetContactModelLists()
{
List<ContactModel> result = new List<ContactModel>(); result.Add(new ContactModel() { PersionName="Tom",PhoneNumber= "" });
result.Add(new ContactModel() { PersionName="HopeGi",PhoneNumber="" });
result.Add(new ContactModel() { PersionName="Jack",PhoneNumber="" });
result.Add(new ContactModel() { PersionName="Tim" }); return result;
}
}
两种模式的差别在于IView和Presenter
PV模式的视图只在提供界面的状态的属性,故IView的接口都是定义属性的
interface IContactView
{
IEnumerable<ContactModel> ListBoxDataSource { set; } ContactModel SelectedContact { get; } string DetailPersonName { set; } string DetailPhoneNumber { set; } event Action SelectedContactChanged;
}
实现IVew接口的Form只需要把传入的属性赋到相应的控件上或者从相应的控件上获取属性返回回去则可。
public partial class Form1 : Form,IContactView
{
public Form1()
{
InitializeComponent();
} public event Action SelectedContactChanged; public IEnumerable<ContactModel> ListBoxDataSource
{
set {
this.lstBoxContact.DataSource = value;
this.lstBoxContact.DisplayMember = "PersionName";
}
} public ContactModel SelectedContact
{
get
{
return this.lstBoxContact.SelectedItem as ContactModel;
}
} public string DetailPersonName
{
set { this.lbDetailPersonName.Text = value; }
} public string DetailPhoneNumber
{
set { this.lbDetailPhoneNumber.Text = value; }
} private void lstBoxContact_SelectedIndexChanged(object sender, EventArgs e)
{
if (SelectedContactChanged != null) SelectedContactChanged();
}
}
在视图中属性的get/set操作只是简单的取值赋值,那么判断逻辑与控制控制就交给了Presenter去处理
class ContactPresenter
{
IContactView view;
ContactModel model; public ContactPresenter(IContactView view)
{
this.view = view;
view.SelectedContactChanged += new Action(view_SelectedContactChanged);
} void view_SelectedContactChanged()
{
ContactModel _model = view.SelectedContact;
if (_model == null) return;
view.DetailPersonName =string.IsNullOrEmpty(_model.PersionName)?"--": _model.PersionName;
view.DetailPhoneNumber =string.IsNullOrEmpty(_model.PhoneNumber)?"--": _model.PhoneNumber;
} public void ShowView()
{
if (view != null && view is Form)
{
view.ListBoxDataSource = ContactModel.GetContactModelLists();
(view as Form).ShowDialog();
}
}
}
那么现在看看另一种的模式——SC模式
主要还是要看IVew的定义,因为IVew能提供什么,Presenter就能调用什么,至于怎么调用那是其次。
interface IContactView
{
void BindListBoxDataSource(IEnumerable<ContactModel> datas);
void BindContactDetail(ContactModel modelDetail);
event Action<ContactModel> SelectedContactChanged;
}
一看上去其实我觉得跟PV模式的接口没太大区别的,因为如果转到Java而言,属性的get/set操作也是通过方法来实现的,这样子只是更换了方法名而已,但是再细心对比一下可以发现有差别,set操作的确换成了用方法来实现而不是用属性,但是这些方法都没有返回值,方法中也没有get操作,因为一般Presenter如果要获取View的参数是,基本上是界面上有用户操作触发了事件,这样Presenter处理用户事件时所需要的数据都用事件参数一并传到Presenter里面,待到Presenter处理完毕后把需要数据展示到View中去。
class ContactPresenter
{
IContactView view;
ContactModel model; public ContactPresenter(IContactView view)
{
this.view = view;
this.view.SelectedContactChanged += new Action<ContactModel>(view_SelectedContactChanged);
} void view_SelectedContactChanged(ContactModel obj)
{
this.view.BindContactDetail(obj);
} public void ShowView()
{
if (view != null && view is Form)
{
this.view.BindListBoxDataSource(ContactModel.GetContactModelLists());
(view as Form).ShowDialog();
}
}
}
这里的Presenter比PV的要简单,因为有一部分代码转移到View中去了
public partial class Form1 : Form,IContactView
{
public Form1()
{
InitializeComponent();
} private void lstBoxContact_SelectedIndexChanged(object sender, EventArgs e)
{
if (SelectedContactChanged != null) SelectedContactChanged(lstBoxContact.SelectedItem as ContactModel);
} public void BindListBoxDataSource(IEnumerable<ContactModel> datas)
{
if (datas == null) return;
lstBoxContact.DataSource = datas;
lstBoxContact.DisplayMember = "PersionName";
} public void BindContactDetail(ContactModel modelDetail)
{
if (modelDetail == null)
{
lbDetailPersonName.Text = "--";
lbDetailPhoneNumber.Text = "--";
}
else if (string.IsNullOrEmpty(modelDetail.PersionName))
lbDetailPersonName.Text = "--";
else if (string.IsNullOrEmpty(modelDetail.PhoneNumber))
lbDetailPhoneNumber.Text = "--";
else
{
lbDetailPersonName.Text=modelDetail.PersionName;
lbDetailPhoneNumber.Text = modelDetail.PhoneNumber;
}
} public event Action<ContactModel> SelectedContactChanged;
}
上述两个例子也模仿了蒋老师书中的Demo,但个人觉得这两个例子也不完全是Model,大概是我用的Model和是把存储数据的Model和处理一些逻辑的Model揉在一起了,因为在ASP.NET MVC中的Model与存储数据用的实体类是有区别的。对SC与PV的理解仅参考了蒋老师书上的一点内容,网上内容不少,如果上面有什么说错的欢迎大家指正,如果大家有更多的参考资料也欢迎分享,谢谢!
MVP的PV模式与SC模式的更多相关文章
- MVP模式和MVVM模式
MVP模式 模型-视图-表示器,也就是MVP模式.是mvc模式的一种衍生模式,专注于改进表示逻辑. 与MVC不同,来自view的调用将委托给presenter(表示器),表示器通过接口与view对话. ...
- js架构设计模式——你对MVC、MVP、MVVM 三种组合模式分别有什么样的理解?
你对MVC.MVP.MVVM 三种组合模式分别有什么样的理解? MVC(Model-View-Controller)MVP(Model-View-Presenter)MVVM(Model-View-V ...
- javascript面向对象基础讲解(工厂模式、构造函数模式、原型模式、混合模式、动态原型模式)
面向对象可以把程序中的关键模块都视为对象,而模块拥有属性及方法.这样我们如果把一些属性及方法封装起来,日后使用将非常方便,也可以避免繁琐重复的工作.接下来将为大家讲解在JS中面向对象的实现. 工厂 ...
- 如何让你的传输更安全——NIO模式和BIO模式实现SSL协议通信
对于SSL/TLS协议,如果要每个开发者都自己去实现显然会带来不必要的麻烦,正是为了解决这个问题Java为广大开发者提供了Java安全套接字扩展--JSSE,它包含了实现Internet安全通信的一系 ...
- 工厂方法模式——创建型模式02
1. 简单工厂模式 在介绍工厂方法模式之前,先介绍一下简单工厂模式.虽然简单工厂模式不属于GoF 23种设计模式,但通常将它作为学习其他工厂模式的入门,并且在实际开发中使用的也较为频繁. (1 ...
- WCF学习之旅—请求与答复模式和单向模式(十九)
一.概述 WCF在通信过程中有三种模式:请求与答复.单向.双工通信.以下我们一一介绍. 二.请求与答复模式 客户端发送请求,然后一直等待服务端的响应(异步调用除外),期间处于假死状态,直到服务端有了答 ...
- Java 策略模式和状态模式
本文是转载的,转载地址:大白话解释Strategy模式和State模式的区别 先上图: 本质上讲,策略模式和状态模式做得是同一件事:去耦合.怎么去耦合?就是把干什么(语境类)和怎么干(策略接口)分开, ...
- Spark运行模式与Standalone模式部署
上节中简单的介绍了Spark的一些概念还有Spark生态圈的一些情况,这里主要是介绍Spark运行模式与Spark Standalone模式的部署: Spark运行模式 在Spark中存在着多种运行模 ...
- Doctype作用?严格模式与混杂模式如何区分?它们有何意义?
怪异模式和严格模式(译注:一般称为标准模式:Standards Mode,下文中的严格模式都可以理解为标准模式)是浏览器解析CSS时的两种‘模式’.这篇文章将简单阐述这两种模式之间的差异. 译注:一个 ...
随机推荐
- C#可扩展编程之MEF学习笔记(五):MEF高级进阶
好久没有写博客了,今天抽空继续写MEF系列的文章.有园友提出这种系列的文章要做个目录,看起来方便,所以就抽空做了一个,放到每篇文章的最后. 前面四篇讲了MEF的基础知识,学完了前四篇,MEF中比较常用 ...
- Word发布博客测试
/// <summary> /// Main相当于Windows服务的Start方法 主要负责初始化的工作 /// </summary> /// <param name= ...
- Redis系列-冷知识
下面是一些看了,但觉得用处不大,不记下又可惜的东西. Redis删除过期数据 redis通过expire/expireat(秒为单位)或者pexpire/pexpireat(毫秒为单位)来设置key的 ...
- IOS Socket 05-XMPP开始&安装服务器openfire&安装配置客户端
1. 即时通讯技术简介(IM) 即时通讯技术(IM-Instant Messageing)支持用户在线实时交谈.如果要发送一条信息,用户需要打开一个小窗口,以便让用户及其朋友在其中输入信息并让交谈双方 ...
- Canvas 内部元素添加事件处理
目录 前言 自定义事件 有序数组 元素父类 事件判断 其他 立即执行函数 apply, call, bind addEventListener 传参 调用父类的构造函数 对象检测 isPointInP ...
- 我心中的核心组件~MSMQ与Redis队列
回到目录 这个文章其实是我心中的核心组件的第七回,确实在时间上有些滞后了,但内容并不滞后!本文MSMQ只是个引题,我确实不太想说它,它是微软自己集成的一套消息队列,寄宿在Window服务里,稳定性十在 ...
- 锋利的JQuery —— Ajax
大图猛戳
- .NET面试题解析(04)-类型、方法与继承
系列文章目录地址: .NET面试题解析(00)-开篇来谈谈面试 & 系列文章索引 做技术是清苦的.一个人,一台机器,相对无言,代码纷飞,bug无情.须梦里挑灯,冥思苦想,肝血暗耗,板凳坐穿 ...
- WPF自定义控件与样式(4)-CheckBox/RadioButton自定义样式
一.前言 申明:WPF自定义控件与样式是一个系列文章,前后是有些关联的,但大多是按照由简到繁的顺序逐步发布的等,若有不明白的地方可以参考本系列前面的文章,文末附有部分文章链接. 本文主要内容: Che ...
- javase基础复习攻略《一》
作为一名软件工程专业,JAVA开发方向的大三本科学生,有必要把自己的学到的专业知识进行一下整理,一则方便自己了解自己的学习程度,二则帮助刚刚接触这一门语言的童鞋了解学习,本人来自与河南理工大学,介绍完 ...

