MVP模式是类似于MVC模式的一种设计模式,最近在做项目学习过程中遇到,弄了很久终于有一些眉目,这是学习过程中的一些笔记。
MVP指的是实体对象Model、视图Viw和业务处理Presenter。MVP的作用是解耦UI渲染、业务逻辑和数据实体的关系。在普通的winform中,业务和界面是写在一起的,一般都是同一个Load或Click方法中,使View和Controller紧密联系。在MVP中,我们将界面渲染放在View里面,也就是winfrom的窗体应用类;将业务关系放在Presenter类中,这就是MVP中的业务数据类;最后的数据实体与数据库的交互放在Model中,是三者各司其职。
一般的MVP中,我们是在Presenter中主动使用View,也就是界面控件形态都是由Presenter去主动控制的。而建立这两者之间的联系,就是在Presenter中注册View的事件,当界面发生由用户触发事件时,这个事件会通过View传递到Presenter中,并在Presenter中调用Model的数据方法,最后Presenter调用在类中引用的View的实例去改变界面形态,下面是一些方法的实现,这里着重说明Presenter和View的关系。
首先,我们定义View的接口类IView,里面就一个数据实体的引用:

 public interface IView<T>:IView
{
T Model { get; set; }
}

public interface IView:IView

接下来,定义View的下一级接口,这里是定义视图一些控件和事件:

 public interface IMainForm<T> : IView<T>
{
Button TestButton { get;} //定义MainFrom的按钮引用
TextBox TestTextBox { get; } //定义MianForm的文本框引用
event EventHandler ViewLoadEvent; //定义窗体加载完毕执行事件
event EventHandler ButtonSubmitEvent; //定义按钮事件
void ShowSubmitDialog(); //定义自定义的事件
}

public interface IMainForm : IView

最后就是View的实现类,里面是实现的接口方法和属性,包含一个按钮及一个文本框,这里有一个继承了的MvpForm类和PresenterBinding的特性,一会再说:

 [PresenterBinding(typeof(MainFormPresenter))]
public partial class MainForm : MvpForm , IMainForm<MainFormModel>
{
public MainForm()
{
InitializeComponent();
} public MainFormModel Model { get; set; }
public TextBox TestTextBox { get { return txtText; } }
public Button TestButton { get { return btnSubmit; } } public event EventHandler ViewLoadEvent;
public event EventHandler ButtonSubmitEvent; private void MainForm_Load(object sender, EventArgs e)
{
if (ViewLoadEvent != null) ViewLoadEvent(sender, e);
} public void ShowSubmitDialog()
{
MessageBox.Show("to submit?");
} private void btnSubmit_Click(object sender, EventArgs e)
{
if (ButtonSubmitEvent != null) ButtonSubmitEvent(sender , e);
}
}

public partial class MainForm : MvpForm , IMainForm

定义完View的内容,就可以看一下Presenter,同样,先有接口,再有实现,先定义Presenter的接口:

 public interface IPresenter<T>:IPresenter where T : class, View.IView
{
T View { get; }
}

public interface IPresenter:IPresenter where T : class, View.IView

这里再定义一个Presenter的抽象类,用于统一各个不同View对应的Presenter类,其定义如下:

 public abstract class Presenter<T> : IPresenter<T> where T : class, View.IView
{
private readonly T view; //这里的view作为引用,用于在presenter中获取View的实例
protected Presenter(T view)
{
this.view = view;
} public T View { get { return view; } }
}

public abstract class Presenter : IPresenter where T : class, View.IView

最后就是对应的View的Presenter类了:

 class MainFormPresenter:Presenter<View.IMainForm<Model.MainFormModel>>
{
public MainFormPresenter(View.IMainForm<Model.MainFormModel> view)
: base(view)
{
view.Model = new Model.MainFormModel(); view.ViewLoadEvent += On_ViewLoad;
view.ButtonSubmitEvent += On_ButtonSubmitClick;
init();
} public void init()
{
//To Do something...
} public void On_ViewLoad(object sender, EventArgs e)
{
//To Do something...
} public void On_ButtonSubmitClick(object sender, EventArgs e)
{
View.ShowSubmitDialog();//通过view的实例调用view的方法来改变控件形态
}
}

class MainFormPresenter:Presenter<View.IMainForm>

这里定义了Presenter和View的接口和实现,下面就是如何将这两个不同的模块联合在一起,这里,使用的是.net的特性和反射。
首先,先建立特性类PresenterBindingAttribute,通过在View的实现类标记该特性类和指定对应的Presenter类,就可以获取View和Presenter的对应关系了,该类同样要标注特性AttributeUsage和继承Attribute类,同时定义两个属性参数:

 [AttributeUsage(AttributeTargets.Class,AllowMultiple = true)]
public sealed class PresenterBindingAttribute : Attribute
{
public Type PresenterType { get;private set; } public Type ViewType { get; set; } public PresenterBindingAttribute(Type presenterType)
{
PresenterType = presenterType;
ViewType = null;
}
}

public sealed class PresenterBindingAttribute : Attribute

接下来,就是通过反射去建立view和presenter的关系,这里建立PerformBinding类:

 public IPresenter PerformBinding(IView viewInstance)
{
IPresenter presenter = null;
Type t = viewInstance.GetType(); //获取该视图的类类型
object[] attrs = t.GetCustomAttributes(typeof(PresenterBindingAttribute), false); //获取该类上的附加特性集合
//遍历特性集合,找到Presenter类型附加的特性,通过该特性建立实例
foreach (PresenterBindingAttribute pba in attrs)
{
Type newt = pba.PresenterType; //获取Presenter类类型
//建立Presenter实例,这里的构造参数是View的对象,这样就使两者建立了联系
Object obj = Activator.CreateInstance(pba.PresenterType, viewInstance);
presenter = obj as IPresenter;
}
return presenter;
}

public IPresenter PerformBinding(IView viewInstance)

那么,这个类PerformBinding在哪里使用,一般是在应用View启动是就要注册实例,这里为了解除类间的强耦合,就添加一个中间类。在前面的View的实现类中,是继承自一个MvpForm的类,这个MvpForm就使注册View和Presenter关系的类,接下来看MvpForm的实现:

 public partial class MvpForm : Form,IView
{
private readonly PresenterBinder presenterBinder = new PresenterBinder();
public MvpForm()
{
presenterBinder.PerformBinding(this); //注册关系
}
}

public partial class MvpForm : Form,IView

这样,就建立了View与Presenter之间的关系,每次View页面启动,就先执行父类MvpForm的构造函数,注册View和Presenter的关系,相应的逻辑可以写在Presenter中,View的作用就是作为UI的渲染。以后添加View和Presenter实现类时,只需要继承和实现相应的类和接口,并在View实现类添加相应的对应类特性,就可实现MVP的设计关系。

c#中winform的MVP模式的简单实现的更多相关文章

  1. Android架构篇--MVP模式的介绍篇

    摘要: 在MVVM成熟之前MVP模式在Android上有被神化的趋势,笔者曾经在商业项目中从零开始大规模采用过MVP模式对项目进行开发.在使用MVP模式进行开发的时候发现项目的结构模式对开发是有一定的 ...

  2. MVP模式入门(结合Rxjava,Retrofit)

    本文MVP的sample实现效果: github地址:https://github.com/xurui1995/MvpSample 老规矩,在说对MVP模式的理解之前还是要再谈谈MVC模式,了解了MV ...

  3. 在Andoid开发中使用MVP模式来解耦,增加可测试性

    by Jeff Angelini posted on 7/20/2011 2:35:00 PM 将应用程序UI的表现从Ui的逻辑中分离是一个好的想法.这种分离减少了代码耦合,代码更加干净, 甚至可以有 ...

  4. Android -- 思考 -- 为什么要在项目中使用MVP模式

    1,其实有时候一直在找借口不去思考这个问题,总是以赶项目为由,没有很认真的思考这个问题,为什么我们要在项目中使用MVP模式,自己也用MVP也已经做了两个项目,而且在网上也看了不少的文章,但是感觉在高层 ...

  5. MVP模式在Android项目中的使用

    以前在写项目的时候,没有过多考虑架构模式的问题,因为之前一直做J2EE开发,而J2EE都是采用MVC模式进行开发的,所以在搭建公司项目的时候,也是使用类似MVC的架构(严格来讲,之前的项目还算不上MV ...

  6. 从最简单的HelloWorld理解MVP模式

    版权声明:本文为博主原创文章,转载请注明出处:http://www.cnblogs.com/joy99/p/6116855.html 大多数编程语言相关的学习书籍,都会以hello,world这个典型 ...

  7. Android 中的MVP 模式

    MVP模式的核心思想: MVP把Activity中的UI逻辑抽象成View接口,把业务逻辑抽象成功接口,Model类还是原来的Model. MVC 其中View层其实就是程序的UI界面,用于向用户展示 ...

  8. MVP模式在Android开发中的应用

    一.MVP介绍      随着UI创建技术的功能日益增强,UI层也履行着越来越多的职责.为了更好地细分视图(View)与模型(Model)的功能,让View专注于处理数据的可视化以及与用户的交互.同一 ...

  9. 浅析前端开发中的 MVC/MVP/MVVM 模式

    MVC,MVP和MVVM都是常见的软件架构设计模式(Architectural Pattern),它通过分离关注点来改进代码的组织方式.不同于设计模式(Design Pattern),只是为了解决一类 ...

随机推荐

  1. 20145305解佳玲 《Java程序设计》第1周学习总结

    教材学习内容总结 第一章 Java平台概论 1.先了解了JAVA的历史 2.Java三大平台:Java SE.Java EE与Java ME 3.Java SE的四个组成部分:JVM.JRE.JDK与 ...

  2. 101、使用ContentProvider在应用间传递数据

    [ContentWriter] package com.jikexueyuan.contentwriter; import android.content.ContentProvider; impor ...

  3. JAVA中关于同步与死锁的问题

    java中当多个现成同时操纵同一资源的时候需要考虑同步的问题.如车站售票,不同售票点卖同一班次车票的时候就要同步,否则卖票会有问题.下面代码模拟车站卖票: class TicketSeller imp ...

  4. lambda表达式、内置函数、进制和文件操作

    lambda表达式 定义函数(普通方式)def f1(): return 123 f2 = lambda : 123 def f3(a1,a2): return a1+a2 定义函数(lambda表达 ...

  5. 视频相关android软件

    1. 视频解码工具:ffmpeg, http://www.ffmpeg.org/ 2. java有一个开源程序: yoyoPlayer, 可以到这个代码中去学习相关的音频知识.http://www.b ...

  6. 玩转JS插件系列

    说明:本系列文章只是通过学习JS插件源码来巩固自己的JS知识,不涉及任何商业目的,如有侵犯版权请尽快告知 一.UI 背景 对话框和灯箱 筛选及排序 反馈 弹出层 悬停 布局 图表 加载 圆边 滚动 标 ...

  7. Android——显示单位px和dip以及sp的区别

    dip: device independent pixels(设备独立像素). 不同设备有不同的显示效果,这个和设备硬件有关,一般我们为了支持WVGA.HVGA和QVGA 推荐使用这个,不依赖像素.d ...

  8. Redis多机功能之Sentinel

    Sentinel的目的:监视主从服务器,并在主服务器下线时自动进行故障转移 启动Sentinel 通过执行Redis安装文件中的redis-sentinel程序,可以启动一个Sentinel实例: r ...

  9. PL/0与Pascal-S编译器程序详细注释

    学校编译课的作业之一,要求阅读两个较为简单的编译器的代码并做注释, 个人感觉是一次挺有意义的锻炼, 将自己的心得分享出来与一同在进步的同学们分享. 今后有时间再做进一步的更新和总结,其中可能有不少错误 ...

  10. 如何查看IIS并发连接数

    如果要想知道确切的当前网站IIS连接数的话,最有效的方法是通过windows自带的系统监视器来查看. 一.运行-->输入"perfmon.msc". 二.在"系统监 ...