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

使用MVP打造项目框架

前言在目前的项目框架中大多是用Viewpager+Fragment实现,而通常情况下一个Fragment中包含以下功能,但是如果将这些功能全部集成在一个fragment中会造成,逻辑不清楚,

前言

在目前的项目框架中大多是用Viewpager+Fragment实现,而通常情况下一个Fragment中包含以下功能,但是如果将这些功能全部集成在一个fragment中会造成,逻辑不清楚,而且我们编写的时候本身也不容易理清顺序,比如在刷新界面的时候要分多种情况,如果是加载第一页且没有缓存数据的时候显示进度动画,否则显示listview自带的下拉刷新动画,当发生错误的时候也要根据有无显示的内容做相应的判断,如果有内容则显示Toast提示用户,否则切换到重试页面。如果将这些逻辑处理与fragment捆绑的话,如果需要更换banner控件,或则下拉刷新控件,则这些逻辑不能复用。所以我尝试使用MVP的方式实现项目框架。

这里写图片描述

功能
先看一下我目前实现的功能有哪些。

1.下拉刷新,上拉加载更多

这里写图片描述

2.懒加载,加载动画,数据缓存

这里写图片描述

3.错误提示(此处有数据,无法显示重试界面,不过如果没有数据的话是有重试界面的)

这里写图片描述

4.下一页重试,(因为已经有了一页数据,所以不能显示重试界面,否则会把用户的数据覆盖,此处添加了一个FooterView用于重试)

这里写图片描述

实现

1.为什么选择MVP

MVP(mode view presenter)与传统的MVC(mode view controller)更注重了view的独立性,在mvc中更注重model的独立性,即model是不变的而有多个view对应于同一个mode,这样就照成了view依赖mode,而view中不可避免的含义逻辑处理的部分,这样将照成view的重用性下降,系统的耦合性提高,而在mvp中mode 与 view 都是独立的。而主要的逻辑都在presenter中,而在presenter中持有的也是model与view的接口,所以这样就使得mode与view的重用性提高,系统的灵活性增加。

我的项目结构如下

这里写图片描述

2.接口抽取

这里面最关键的是对view接口的抽取,以前也看过许多人的mvpdemo但是等到自己写的时候却不知道要怎么办,我总结了一下对view接口抽取主要是三点

  1. 需要从view中获取的数据
  2. 需要让view显示的状态
  3. 需要让view执行的操作

IListView借口

package com.zgh.mvpdemo.view.news;

import android.content.DialogInterface;

import com.zgh.mvpdemo.bean.BannerItem;
import com.zgh.mvpdemo.bean.NewsItem;

import java.util.List;

/**
* Created by zhuguohui on 2016/6/29.
*/

public interface IListView {
/*******加载首页相关*******/

//显示加载首页的加载效果
void showLoadingFirstPage();

//隐藏加载首页动画
void hideLoadingFirstPage();

//首页加载失败的时候调用
void showRetryFirstPage();

//首页为空的时候调用
void showEmpty();

//获取到首页数据的时候调用
void showFirstPageData(List listData);


/***********加载下一页相关*************/

//显示加载下一页
void showLoadingNextPage();

//隐藏加载下一页
void hideLoadingNextPage();

//获取到下一页数据是调用
void showNextPageData(List listData);

//显示还有下一页
void showHaveNextPage();

//显示重试加载下一页
void showRetryNextPage();

//没有更多了
void showNoMore();

/**************Banner相关*********************/

//隐藏banner在没有banner数据的时候调用
void hideBanner();

//显示banner数据
void showBanner(List bannerData);


/*****************获取数据***************************/

//判断是否还有下一页
boolean haveNextPage(int dataSize);

//是否已经有显示的内容了,用于判断在没有网络的时候是否显示重试界面,如果有内容则显示内容,否则显示重试。
boolean haveContent();

//获取首页的url地址
String getFirstPageUrl();

//下一页的url地址
String getNextPageUrl();

/**************点击事件****************************/

interface OnBannerClickListener {
void onBannnerClick(BannerItem item);
}

interface OnNewsItemClickListener {
void onNewsItemClick(NewsItem item);
}

void setOnBannerItemClickListener(OnBannerClickListener listener);

void setOnListItemClickListener(OnNewsItemClickListener listener);

void toItemDetail(NewsItem item);

void toBannerDetail(BannerItem item);


/****************通知*******************************/

void showToast(String info);

}

IListMode 接口

package com.zgh.mvpdemo.model.news;

/**
* Created by zhuguohui on 2016/6/29.
*/

public interface IListMode {
/**
*
* @param useCache 是否使用缓存
* @param url 数据url
* @param listener 回调接口
*/

void LoadData(boolean useCache,String url,DataResultListener listener);
}

对Presenter就没有抽取接口了

package com.zgh.mvpdemo.presenter.news;

import com.zgh.mvpdemo.bean.BannerItem;
import com.zgh.mvpdemo.bean.NewsItem;
import com.zgh.mvpdemo.model.news.DataResultListener;
import com.zgh.mvpdemo.model.news.IListMode;
import com.zgh.mvpdemo.model.news.ListMode;
import com.zgh.mvpdemo.view.news.IListView;

import java.util.List;

/**
* Created by zhuguohui on 2016/6/29.
*/

public class ListPresenter implements IListView.OnBannerClickListener, IListView.OnNewsItemClickListener {
IListView listView;
IListMode listMode;

public ListPresenter(IListView listView) {
this.listView = listView;

listMode = new ListMode();
//设置点击事件
listView.setOnBannerItemClickListener(this);
listView.setOnListItemClickListener(this);
}

public void LoadFirstPage(boolean useCache) {
//在没有内容的时候才显示进度条,在下拉刷新的时候不需要
if(!listView.haveContent()) {
listView.showLoadingFirstPage();
}
listMode.LoadData(useCache,listView.getFirstPageUrl(), new DataResultListener() {
@Override
public void onSuccess(List newsData, List bannerData) {
listView.hideLoadingFirstPage();
//如果banner数据不为空才显示banner,否则隐藏
if (bannerData != null && bannerData.size() > 0) {
listView.showBanner(bannerData);
} else {
listView.hideBanner();
}
//根据list是否有数据设置显示样式
if (newsData != null && newsData.size() > 0) {
listView.showFirstPageData(newsData);
//根据item的数量判断是否有下一页
if (listView.haveNextPage(newsData.size())) {
listView.showHaveNextPage();
} else {
listView.showNoMore();
}
} else {
listView.showEmpty();
}

}

@Override
public void onError(String error) {
listView.hideLoadingFirstPage();
//在没有内容的时候才显示重试,否则只提示,还是显示原来的缓存内容
if (!listView.haveContent()) {
listView.showRetryFirstPage();
}
listView.showToast(error);
}

});
}

public void LoadNextPage() {
//显示加载下一页
listView.showLoadingNextPage();
listMode.LoadData(false,listView.getNextPageUrl(), new DataResultListener() {
@Override
public void onSuccess(List newsData, List bannerData) {
//在加载下一页的时候不需要判断banner的数据

if (newsData != null && newsData.size() > 0) {
listView.showNextPageData(newsData);
//如果没有下一页,则显示没有更多了
if (!listView.haveNextPage(newsData.size())) {
listView.showNoMore();
}else{
listView.showHaveNextPage();
}
} else {
listView.showToast("没有更多了");
listView.showNoMore();
}
listView.hideLoadingFirstPage();
}

@Override
public void onError(String error) {
listView.hideLoadingNextPage();
//在没有内容的时候才显示重试,否则只提示,还是显示原来的缓存内容
if (!listView.haveContent()) {
listView.showRetryFirstPage();
}else{
listView.showRetryNextPage();
}
listView.showToast(error);
}

});
}

@Override
public void onBannnerClick(BannerItem item) {
listView.toBannerDetail(item);
}

@Override
public void onNewsItemClick(NewsItem item) {
listView.toItemDetail(item);
}
}

3.状态切换

状态切换我使用的是张鸿洋的LoadingAndRetryManager 关于具体的用法大家自己去看吧。需要说明的是,LoadingAndRetryManager在ViewPager中的fragment中使用的时候要这么用,要将LoadingAndRetryLayout作为fragment的view返回,而第一个mBaseView是你自己创建的view。

    mLoadingAndRetryManager = new LoadingAndRetryManager(mBaseView, new OnLoadingAndRetryListener() {
@Override
public void setRetryEvent(View retryView) {
retryView.findViewById(R.id.id_btn_retry).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
mPresenter.LoadFirstPage(true);
}
});
}
});
mBaseView = mLoadingAndRetryManager.mLoadingAndRetryLayout;

4.Fragment与ViewPager使用时的注意事项

1.懒加载

由于viewpager会预先缓存几页Fragment,所以Fragment的生命周期在ViewPager中其实是没有多少意义的,因此为了实现用户滑动到这个界面才显示这个界面的数据的功能,我们必须在setUserVisibleHint中加载数据,就像这样

    @Override
public void setUserVisibleHint(boolean isVisibleToUser) {
super.setUserVisibleHint(isVisibleToUser);
if (isVisibleToUser && !haveInint) {
haveInint = true;
mPresenter.LoadFirstPage(true);
}
}

View的缓存

在viewpager的滑动过程中,某个fragment的onCreateView()方法和onDestroyView()方法会多次调用,也就是会先销毁view而保留Fragment中的成员变量,等内存不足是再销毁Fragment。由于View的创建也是很耗时的操作,所以我选择缓存view,而且为了在数据请求返回的时候view都存在,防止每次都有判断view是否为空,因此我将view的创建放在了onCreate()方法中,且先创建view再请求数据。

   @Override
public void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
createView();
bindData();
setListener();
mPresenter = new ListPresenter(this);
mLoadingAndRetryManager.showLoading();
}


@Nullable
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
ViewParent parent = mBaseView.getParent();
if (parent != null && parent instanceof ViewGroup) {
ViewGroup group = (ViewGroup) parent;
group.removeView(mBaseView);
}
return mBaseView;
}

5.ListView的FooterView与HeardView的显示与隐藏

我觉得这个技巧还是蛮有用的,于是专门写一下。在很多时候我们将view添加到ListView中作为FootView或者HeardView,想隐藏的时候就调用view.setVisibility(View.GONE),然而结果是view的确不显示了,但是它所占用的空间还在,与调用view.setVisibility(View.INVISIBLE)效果类似,后来我发现在添加footview或headview之前用一层layout进行包裹就可以实现隐藏footerview的效果了,代码如下:

//添加footerview
tv_bottom_info = (TextView) View.inflate(getActivity(), R.layout.view_bottom_retry, null);
tv_bottom_info.setLayoutParams(new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, DensityUtil.dip2px(getActivity(), 45)));
//要想footerview实现隐藏效果,必须在其外部包裹一层layout,heardview 同理
LinearLayout footerviewParent = new LinearLayout(getActivity());
footerviewParent.addView(tv_bottom_info);
tv_bottom_info.setVisibility(View.GONE);
mListView.addFooterView(footerviewParent);

Demo

更多细节请看我的demo,欢迎star哈。

MVPDemo


推荐阅读
  • Android开发实现的计时器功能示例
    本文分享了Android开发实现的计时器功能示例,包括效果图、布局和按钮的使用。通过使用Chronometer控件,可以实现计时器功能。该示例适用于Android平台,供开发者参考。 ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • Go GUIlxn/walk 学习3.菜单栏和工具栏的具体实现
    本文介绍了使用Go语言的GUI库lxn/walk实现菜单栏和工具栏的具体方法,包括消息窗口的产生、文件放置动作响应和提示框的应用。部分代码来自上一篇博客和lxn/walk官方示例。文章提供了学习GUI开发的实际案例和代码示例。 ... [详细]
  • Go Cobra命令行工具入门教程
    本文介绍了Go语言实现的命令行工具Cobra的基本概念、安装方法和入门实践。Cobra被广泛应用于各种项目中,如Kubernetes、Hugo和Github CLI等。通过使用Cobra,我们可以快速创建命令行工具,适用于写测试脚本和各种服务的Admin CLI。文章还通过一个简单的demo演示了Cobra的使用方法。 ... [详细]
  • 数组的排序:数组本身有Arrays类中的sort()方法,这里写几种常见的排序方法。(1)冒泡排序法publicstaticvoidmain(String[]args ... [详细]
  • (三)多表代码生成的实现方法
    本文介绍了一种实现多表代码生成的方法,使用了java代码和org.jeecg框架中的相关类和接口。通过设置主表配置,可以生成父子表的数据模型。 ... [详细]
  • Android系统源码分析Zygote和SystemServer启动过程详解
    本文详细解析了Android系统源码中Zygote和SystemServer的启动过程。首先介绍了系统framework层启动的内容,帮助理解四大组件的启动和管理过程。接着介绍了AMS、PMS等系统服务的作用和调用方式。然后详细分析了Zygote的启动过程,解释了Zygote在Android启动过程中的决定作用。最后通过时序图展示了整个过程。 ... [详细]
  • 本文讨论了如何优化解决hdu 1003 java题目的动态规划方法,通过分析加法规则和最大和的性质,提出了一种优化的思路。具体方法是,当从1加到n为负时,即sum(1,n)sum(n,s),可以继续加法计算。同时,还考虑了两种特殊情况:都是负数的情况和有0的情况。最后,通过使用Scanner类来获取输入数据。 ... [详细]
  • 《数据结构》学习笔记3——串匹配算法性能评估
    本文主要讨论串匹配算法的性能评估,包括模式匹配、字符种类数量、算法复杂度等内容。通过借助C++中的头文件和库,可以实现对串的匹配操作。其中蛮力算法的复杂度为O(m*n),通过随机取出长度为m的子串作为模式P,在文本T中进行匹配,统计平均复杂度。对于成功和失败的匹配分别进行测试,分析其平均复杂度。详情请参考相关学习资源。 ... [详细]
  • [大整数乘法] java代码实现
    本文介绍了使用java代码实现大整数乘法的过程,同时也涉及到大整数加法和大整数减法的计算方法。通过分治算法来提高计算效率,并对算法的时间复杂度进行了研究。详细代码实现请参考文章链接。 ... [详细]
  • Android源码深入理解JNI技术的概述和应用
    本文介绍了Android源码中的JNI技术,包括概述和应用。JNI是Java Native Interface的缩写,是一种技术,可以实现Java程序调用Native语言写的函数,以及Native程序调用Java层的函数。在Android平台上,JNI充当了连接Java世界和Native世界的桥梁。本文通过分析Android源码中的相关文件和位置,深入探讨了JNI技术在Android开发中的重要性和应用场景。 ... [详细]
  • Java自带的观察者模式及实现方法详解
    本文介绍了Java自带的观察者模式,包括Observer和Observable对象的定义和使用方法。通过添加观察者和设置内部标志位,当被观察者中的事件发生变化时,通知观察者对象并执行相应的操作。实现观察者模式非常简单,只需继承Observable类和实现Observer接口即可。详情请参考Java官方api文档。 ... [详细]
  • Spring学习(4):Spring管理对象之间的关联关系
    本文是关于Spring学习的第四篇文章,讲述了Spring框架中管理对象之间的关联关系。文章介绍了MessageService类和MessagePrinter类的实现,并解释了它们之间的关联关系。通过学习本文,读者可以了解Spring框架中对象之间的关联关系的概念和实现方式。 ... [详细]
  • 面向对象之3:封装的总结及实现方法
    本文总结了面向对象中封装的概念和好处,以及在Java中如何实现封装。封装是将过程和数据用一个外壳隐藏起来,只能通过提供的接口进行访问。适当的封装可以提高程序的理解性和维护性,增强程序的安全性。在Java中,封装可以通过将属性私有化并使用权限修饰符来实现,同时可以通过方法来访问属性并加入限制条件。 ... [详细]
author-avatar
能豆子314
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有