热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

MVP架构模式及其在国庆技术博客中的应用

本文介绍了MVP架构模式及其在国庆技术博客中的应用。MVP架构模式是一种演变自MVC架构的新模式,其中View和Model之间的通信通过Presenter进行。相比MVC架构,MVP架构将交互逻辑放在Presenter内部,而View直接从Model中读取数据而不是通过Controller。本文还探讨了MVP架构在国庆技术博客中的具体应用。

                                        MVP架构模式

                   我的意识里Mvp相当于MainActivity找了个秘书,在 Presenter操作,最终决定权还是在MainActivity里,我个人是这样理解的,然后就是网上的一些官方的描述,


       MVP是单词Model View Presenter的首字母的缩写,分别表示数据层、视图层、发布层,它是MVC架构的一种演变。作为一种新的模式,MVP与MVC有着一个重大的区别:

 在MVP中View并不直接使用Model,它们之间的通信是通过Presenter (MVC中的Controller)来进行的,所有的交互都发生在Presenter内部,而在MVC中View会直接从Model中读取数据而不是通过 Controller。
首先我们先看下传统的MVC架构Model View Controller,我们把业务逻辑放到C层(ios的ViewController,android的Activity&Fragment),但是这里会引入另外一个问题,所有的逻辑都在C层,不可避免的会造成C层非常复杂,如果项目越来越大,C层的代码会更加臃肿,维护起来也非常麻烦,而且也没办法==简单的==做单元测试,试想做一个单元测试我们要加入多少逻辑代码?

     

   现有的MVC模式存在以下问题:

  1. 视图与控制器间的过于紧密的连接

    视图与控制器是相互分离,但却是联系紧密的部件,视图没有控制器的存在,其应用是很有限的,反之亦然,这样就妨碍了他们的独立重用。

  2. 视图对模型数据的低效率访问

    依据模型操作接口的不同,视图可能需要多次调用才能获得足够的显示数据。对未变化数据的不必要的频繁访问,也将损害操作性能。

     MVP
1,  MVP分离了view和model层,Presenter层充当了桥梁的角色,View层只负责更新界面即可,这里的View我们要明白只是一个viewinterface,它是视图的接口,这样我们在做单元测试的时候可以非常方便编写Presenter层代码。
2,厚重的Controller层代码也得到了释放,之前我们开发的时候会对UIViewController、Activity、Fragment编写很多的业务逻辑,尽管大家会将Service层做分离,如net层,DB层等,但还是无法避免类似的问题,activity uicontroller无法重复利用是非常难以忍受的。
3,有一点还需要注意,presenter是双向绑定的关系,因此,在设计的时候就要注意接口和抽象的使用,尽可能的降低代码的耦合度,这也是mvp的宗旨。

转向mvp吧!我们先看下MVP几个单词的意思,以下是我个人的理解:

  • View: 是显示数据(model)并且将用户指令(events)传送到presenter以便作用于那些数据的一个接口。View通常含有Presenter的引用。在Android开发中通常将Activity或者Fragment作为View层。
  • Model: 对于Model层也是数据层。它区别于MVC架构中的Model,在这里不仅仅只是数据模型。在MVP架构中Model它负责对数据的存取操作,例如对数据库的读写,网络的数据的请求等。
  • Presenter:对于Presenter层他是连接View层与Model层的桥梁并对业务逻辑进行处理。在MVP架构中Model与View无法直接进行交互。所以在Presenter层它会从Model层获得所需要的数据,进行一些适当的处理后交由View层进行显示。这样通过Presenter将View与Model进行隔离,使得View和Model之间不存在耦合,同时也将业务逻辑从View中抽离。
      示例
 下面是项目的架构,一个Activity,一个Fragment,Data层主要负责获取App已安装的应用列表,AppListPresenter负责业务逻辑处理
   
  • AppListFragment的代码

    public class AppListFragment extends Fragment implements AppViewInterface {private Presenter presenter;private List packageInfoList &#61; new ArrayList<>();private RecyclerView recyclerView;private MyAppListRecyclerViewAdapter myAppListRecyclerViewAdapter;&#64;Overridepublic void showAppList(List packageInfos) {if (packageInfos.isEmpty())return;packageInfoList.clear();packageInfoList.addAll(packageInfos);myAppListRecyclerViewAdapter.notifyDataSetChanged();}&#64;Overridepublic void setPresenter(Presenter presenter) {this.presenter &#61; presenter;}}

    代码比较容易理解&#xff0c;AppListFragment实现了AppViewInterface接口&#xff0c;我们需要在Activity中把AppListPresenter和AppViewInterface双向绑定。

  • 接下来看下AppListPresenter层的代码&#xff0c;这里只列出了几个关键方法

    public class AppListPresenter implements Presenter, LoaderManager.LoaderCallbacks<List<PackageInfo>>{private AppViewInterface viewInterface;private AppClassLoader appClassLoader;private LoaderManager loaderManager;private final int id &#61; 0;public AppListPresenter(AppViewInterface viewInterface, AppClassLoader appClassLoader, LoaderManager loaderManager) {this.viewInterface &#61; viewInterface;this.appClassLoader &#61; appClassLoader;this.loaderManager &#61; loaderManager;viewInterface.setPresenter(this);}&#64;Overridepublic void loadInstallApps() {//通过loadmanager提供的方法获取安装的应用列表loaderManager.initLoader(id, null, this);}&#64;Overridepublic void destory() {loaderManager.destroyLoader(id);}&#64;Overridepublic void onLoadFinished(Loader> loader, List data) {//获取到已安装的应用列表&#xff0c;调用AppViewInterface的showAppList方法viewInterface.showAppList(data);}&#64;Overridepublic void launchApp(PackageInfo packageInfo) {Intent intent &#61; appClassLoader.queryLaunchIntent(packageInfo);if (intent !&#61; null)appClassLoader.getContext().startActivity(intent);elseToast.makeText(appClassLoader.getContext(), "Can not start the app", Toast.LENGTH_SHORT).show();}}

    关键方法是loadInstallApps&#xff0c;这个方法在MainActivity的onCreat中调用

    private Presenter appListPresenter;&#64;Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Toolbar toolbar &#61; (Toolbar) findViewById(R.id.toolbar);setSupportActionBar(toolbar);FragmentTransaction fragmentTransaction &#61; getSupportFragmentManager().beginTransaction();AppListFragment appListFragment &#61; AppListFragment.newInstance();fragmentTransaction.add(R.id.fm, appListFragment);fragmentTransaction.commit();appListPresenter &#61; new AppListPresenter(appListFragment, new AppClassLoader(getApplicationContext()), getSupportLoaderManager());//调用loadInstallAppsappListPresenter.loadInstallApps();}

    首先&#xff0c;我们获取一个AppListFragment的实例&#xff0c;在AppListPresenter构造函数里面我们传入AppViewInterface&#xff0c;同时在AppPresenter的构造函数中又将presenter注入到了AppViewInerface里面&#xff0c;这样就实现了Presenter和ViewInerface双向绑定&#xff0c;之后调用AppPresenter的loadInstallApps方法&#xff0c;在onLoadFinished回调里面又调用了AppViewInterface的showApps方法&#xff0c;这样数据就显示在界面。整个Activity和Fragment的代码精简了很多。

     

       缺点

         由于对视图的渲染放在了Presenter中&#xff0c;所以视图和Presenter的交互会过于频繁。还有一点需要明白&#xff0c;如果Presenter过多地渲染了视图&#xff0c;往往会使得它与特定的视图的联系过于紧密。一旦视图需要变更&#xff0c;那么Presenter也需要变更了



 
 



推荐阅读
  • iOS Swift中如何实现自动登录?
    本文介绍了在iOS Swift中如何实现自动登录的方法,包括使用故事板、SWRevealViewController等技术,以及解决用户注销后重新登录自动跳转到主页的问题。 ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • 欢乐的票圈重构之旅——RecyclerView的头尾布局增加
    项目重构的Git地址:https:github.comrazerdpFriendCircletreemain-dev项目同步更新的文集:http:www.jianshu.comno ... [详细]
  • Android系统源码分析Zygote和SystemServer启动过程详解
    本文详细解析了Android系统源码中Zygote和SystemServer的启动过程。首先介绍了系统framework层启动的内容,帮助理解四大组件的启动和管理过程。接着介绍了AMS、PMS等系统服务的作用和调用方式。然后详细分析了Zygote的启动过程,解释了Zygote在Android启动过程中的决定作用。最后通过时序图展示了整个过程。 ... [详细]
  • 使用nodejs爬取b站番剧数据,计算最佳追番推荐
    本文介绍了如何使用nodejs爬取b站番剧数据,并通过计算得出最佳追番推荐。通过调用相关接口获取番剧数据和评分数据,以及使用相应的算法进行计算。该方法可以帮助用户找到适合自己的番剧进行观看。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • SpringBoot uri统一权限管理的实现方法及步骤详解
    本文详细介绍了SpringBoot中实现uri统一权限管理的方法,包括表结构定义、自动统计URI并自动删除脏数据、程序启动加载等步骤。通过该方法可以提高系统的安全性,实现对系统任意接口的权限拦截验证。 ... [详细]
  • Java实战之电影在线观看系统的实现
    本文介绍了Java实战之电影在线观看系统的实现过程。首先对项目进行了简述,然后展示了系统的效果图。接着介绍了系统的核心代码,包括后台用户管理控制器、电影管理控制器和前台电影控制器。最后对项目的环境配置和使用的技术进行了说明,包括JSP、Spring、SpringMVC、MyBatis、html、css、JavaScript、JQuery、Ajax、layui和maven等。 ... [详细]
  • Mac OS 升级到11.2.2 Eclipse打不开了,报错Failed to create the Java Virtual Machine
    本文介绍了在Mac OS升级到11.2.2版本后,使用Eclipse打开时出现报错Failed to create the Java Virtual Machine的问题,并提供了解决方法。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • 本文介绍了iOS数据库Sqlite的SQL语句分类和常见约束关键字。SQL语句分为DDL、DML和DQL三种类型,其中DDL语句用于定义、删除和修改数据表,关键字包括create、drop和alter。常见约束关键字包括if not exists、if exists、primary key、autoincrement、not null和default。此外,还介绍了常见的数据库数据类型,包括integer、text和real。 ... [详细]
  • Java自带的观察者模式及实现方法详解
    本文介绍了Java自带的观察者模式,包括Observer和Observable对象的定义和使用方法。通过添加观察者和设置内部标志位,当被观察者中的事件发生变化时,通知观察者对象并执行相应的操作。实现观察者模式非常简单,只需继承Observable类和实现Observer接口即可。详情请参考Java官方api文档。 ... [详细]
  • (三)多表代码生成的实现方法
    本文介绍了一种实现多表代码生成的方法,使用了java代码和org.jeecg框架中的相关类和接口。通过设置主表配置,可以生成父子表的数据模型。 ... [详细]
  • MVC设计模式的介绍和演化过程
    本文介绍了MVC设计模式的基本概念和原理,以及在实际项目中的演化过程。通过分离视图、模型和控制器,实现了代码的解耦和重用,提高了项目的可维护性和可扩展性。详细讲解了分离视图、分离模型和分离控制器的具体步骤和规则,以及它们在项目中的应用。同时,还介绍了基础模型的封装和控制器的命名规则。该文章适合对MVC设计模式感兴趣的读者阅读和学习。 ... [详细]
author-avatar
发的好地方
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有