Android应用架构之Android MVP使用
前两篇已经将Retrofit和RxAndroid应用到了项目中,这篇本打算直接将Dagger2引进项目,但是考虑到整个项目结构,就来个结构整理吧,一起来看看网上炒得火热MVP模式。
说到MVP就不得不提到MVC,做过J2EE的猿友们肯定知道MVC是个什么东西。MVC即 Model、View、Controller, 那MVP就Model、View、Presenter。Model用于提供数据模型,View用于显示数据,当然Presenter也就用来处理业务逻辑并将数据显示数据到View上了,它是Model和View的桥梁。
想更清晰的理解MPV如何工作,那就直接上代码吧。
我将整个项目分为三个Module,如下图
一、数据处理模块domain(包含Model),这里的domain命名只是我喜欢这样把数据处理相关的东西都放在这里。
ServiceManager用于向外提供数据的入口(其他类代码在前面博文中已提及)
package com.micky.retrofitrxandroiddragger2.domain.service;
import retrofit.GsonConverterFactory;
import retrofit.Retrofit;
import retrofit.RxJavaCallAdapterFactory;
/**
* @Project RetrofitRxAndroidDragger2
* @Packate com.micky.retrofitrxandroiddragger2.domain.service
* @Description
* @Author Micky Liu
* @Email mickyliu@126.com
* @Date 2015-12-22 14:43
* @Version 1.0
*/
public class ServiceManager {
private static final String ENDPOINT = "http://ip.taobao.com";
private static class ServiceManagerHolder {
private static final ServiceManager INSTANCE = new ServiceManager();
}
private ServiceManager() {}
public static final ServiceManager getInstance() {
return ServiceManagerHolder.INSTANCE;
}
private ApiService mApiService = null;
public ApiService getApiService() {
if (mApiService == null) {
Retrofit retrofit = new Retrofit.Builder()
.baseUrl(ENDPOINT)
.addConverterFactory(GsonConverterFactory.create())
.addCallAdapterFactory(RxJavaCallAdapterFactory.create())
.build();
mApiService = retrofit.create(ApiService.class);
return mApiService;
}
return mApiService;
}
}
二、View和Presenter模块
为了方便管理我将View和Presenter相关类都放在Presentation模块中
View接口
/**
* @Project RetrofitRxAndroidDragger2
* @Packate com.micky.presentation
* @Description
* @Author Micky Liu
* @Email mickyliu@126.com
* @Date 2015-12-22 13:55
* @Version 1.0
*/
public interface MainView {
void showProgress();
void hideProgress();
void setIpText(String text);
}
Presenter接口
package com.micky.retrofitrxandroiddragger2.presenter;
/**
* @Project RetrofitRxAndroidDragger2
* @Packate com.micky.presentation
* @Description
* @Author Micky Liu
* @Email mickyliu@126.com
* @Date 2015-12-22 13:46
* @Version 1.0
*/
public interface MainPresenter extends BasePresenter {
void getIpInfo(String ip);
}
Presenter实现类
package com.micky.retrofitrxandroiddragger2.presenter.impl;
import android.text.TextUtils;
import android.util.Log;
import android.widget.Toast;
import com.micky.retrofitrxandroiddragger2.BaseApplication;
import com.micky.retrofitrxandroiddragger2.R;
import com.micky.retrofitrxandroiddragger2.domain.service.ServiceManager;
import com.micky.retrofitrxandroiddragger2.domain.service.response.GetIpInfoResponse;
import com.micky.retrofitrxandroiddragger2.presenter.MainPresenter;
import com.micky.retrofitrxandroiddragger2.presenter.impl.BasePresenterImpl;
import com.micky.retrofitrxandroiddragger2.ui.view.MainView;
import rx.Subscriber;
import rx.android.schedulers.AndroidSchedulers;
import rx.schedulers.Schedulers;
/**
* @Project RetrofitRxAndroidDragger2
* @Packate com.micky.retrofitrxandroiddragger2.presenter
* @Description
* @Author Micky Liu
* @Email mickyliu@126.com
* @Date 2015-12-22 14:33
* @Version 1.0
*/
public class MainPresenterImpl extends BasePresenterImpl implements MainPresenter {
private static final String TAG = "TAG";
private MainView mMainView;
public MainPresenterImpl(MainView mainView) {
mMainView = mainView;
}
@Override
public void getIpInfo(String ip) {
if (TextUtils.isEmpty(ip)) {
Toast.makeText(BaseApplication.getContext(), R.string.input_tip_ip, Toast.LENGTH_SHORT).show();
return;
}
mMainView.setIpText("");
mMainView.showProgress();
ServiceManager.getInstance().getApiService().getIpInfo(ip)
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new Subscriber<GetIpInfoResponse>() {
@Override
public void onCompleted() {
mMainView.hideProgress();
}
@Override
public void onError(Throwable e) {
Log.e(TAG, e.getMessage(), e);
mMainView.hideProgress();
mMainView.setIpText(BaseApplication.getContext().getString(R.string.network_error));
}
@Override
public void onNext(GetIpInfoResponse getIpInfoResponse) {
mMainView.setIpText(getIpInfoResponse.data.country + " " + getIpInfoResponse.data.area);
}
});
}
}
MainActivity
package com.micky.retrofitrxandroiddragger2.ui.activity;
import android.os.Bundle;
import android.support.design.widget.FloatingActionButton;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.Toolbar;
import android.view.View;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.EditText;
import android.widget.ProgressBar;
import android.widget.TextView;
import com.micky.retrofitrxandroiddragger2.R;
import com.micky.retrofitrxandroiddragger2.presenter.MainPresenter;
import com.micky.retrofitrxandroiddragger2.presenter.impl.MainPresenterImpl;
import com.micky.retrofitrxandroiddragger2.ui.view.MainView;
/**
* @Project RetrofitRxAndroidDragger2
* @Packate com.micky.presentation
* @Description
* @Author Micky Liu
* @Email mickyliu@126.com
* @Date 2015-12-22 12:22
* @Version 1.0
*/
public class MainActivity extends AppCompatActivity implements MainView {
private EditText mEtIp;
private TextView mTvContent;
private ProgressBar mProgressBar;
private MainPresenter mMainPresenter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(toolbar);
mEtIp = (EditText) findViewById(R.id.et_ip);
mTvContent = (TextView) findViewById(R.id.tv_content);
mProgressBar = (ProgressBar) findViewById(R.id.progress_bar);
mMainPresenter = new MainPresenterImpl(this);
FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
fab.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
mMainPresenter.getIpInfo(mEtIp.getText().toString());
}
});
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate(R.menu.menu_main, menu);
return true;
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
int id = item.getItemId();
if (id == R.id.action_settings) {
return true;
}
return super.onOptionsItemSelected(item);
}
@Override
public void showProgress() {
mProgressBar.setVisibility(View.VISIBLE);
}
@Override
public void hideProgress() {
mProgressBar.setVisibility(View.GONE);
}
@Override
public void setIpText(String text) {
mTvContent.setText(text);
}
}
OK,代码基本完了,看了上述代码大家也许都会说以前就在一个类里面就搞定的功能,现在怎么多出这么多接口、实现类啊。别急我刚看到这个mvp的时候也这么想,几度写着写着都把它抛之脑后,这类都多得看不过来了。
仔细研究下不难发现这M、V、P三者的关系还是挺清晰的。
为了让大家更清晰的理解,请看如下时序图(MainView只是为了更好的展示调用与数据流程而特意画出来的)
看到这里,是不是觉得so simple! 呢。
详细代码在老地方: https://github.com/mickyliu945/CommonProj
本文转载自:http://blog.csdn.net/liuhongwei123888/article/details/50380368。
Android应用架构之Android MVP使用的更多相关文章
- 【转】Android官方架构项目之MVP + Clean
首先,不了解 Clean 架构的可以看看这个,不过也没关系,阅读本文后你也许会对Clean架构思想有一个认识. 对比MVP项目的结构图,我们发现不同之处是新增的这个Domain Layer这层,来隔离 ...
- (转)android系统架构及源码目录结构
转自:http://blog.csdn.net/finewind/article/details/46324507 1. Android系统架构: android系统架构采用了分层架构的思想,如下图所 ...
- Android App的设计架构:MVC,MVP,MVVM与架构经验谈
相关:http://www.cnblogs.com/wytiger/p/5996876.html 和MVC框架模式一样,Model模型处理数据代码不变在Android的App开发中,很多人经常会头疼于 ...
- Android 架构艺术之MVP
MVP是Google官方发布的Android开发相关的架构知识.本文要讲解的是一种最基本的MVP的实现方式,它使用手动的依赖注入来提供具有本地和远程数据源的存储库.异步任务处理回调. 基本的MVP的项 ...
- Android 程序架构: MVC、MVP、MVVM、Unidirectional、Clean...
摘选自:GUI 应用程序架构的十年变迁:MVC.MVP.MVVM.Unidirectional.Cleanhttps://zhuanlan.zhihu.com/p/26799645 MV* in An ...
- Android APP架构设计——MVP的使用示例
0. 前言 为了更好地进行移动端架构设计,我们最常用的就是MVC.MVP和MVVM,作为三个最耳熟能详的三大架构,应用可谓非常广泛.对于这三种架构设计以及优缺点已经在Android APP架构设计-- ...
- Android架构(一)MVP架构在Android中的实践
Android架构(一)MVP架构在Android中的实践 https://www.300168.com/yidong/show-2790.html 核心提示:为什么要重视程序的架构设计 对程序进 ...
- Android App的设计架构:MVC,MVP,MVVM与架构AAAAA
1. 架构设计的目的1.1 通过设计使程序模块化,做到模块内部的高聚合和模块之间的低耦合.1.2 这样做的好处是使得程序在开发的过程中,开发人员只需要专注于一点,提高程序开发的效率,并且更容易进行后续 ...
- 从零开始搭建架构实施Android项目
我们先假设一个场景需求:刚有孩子的爸爸妈妈对用照片.视频记录宝宝成长有强烈的意愿,但苦于目前没有一款专门的手机APP做这件事.A公司洞察到市场需求,要求开发团队尽快完成Android客户端的开发.以下 ...
随机推荐
- 多线程同步工具——CAS原子变量
这是我参考的一篇文章<基于CAS的乐观锁实现>,讲述的是一种需要CPU支持的执行技术CAS(Compare and Swap). 首先理解什么是原子性操作,意思是不能再拆分的操作,例如改写 ...
- Apworks框架实战
Apworks框架实战(一):Apworks到底是什么? Apworks框架实战(二):开始使用 Apworks框架实战(三):单元测试与持续集成 Apworks框架实战(四):使用Visual St ...
- Unity iOS混合开发界面切换思路
Unity iOS混合开发界面切换思路 最近有很多博友QQ 私信 或则 留言联系我,请教iOS和Unity界面之前相互切换的问题,源代码就不私下发你们了,界面跳转功能的代码我直接贴到下面好了,顺带说i ...
- EntityFramework中的DbContext使用疑点说明
1.DbContext怎么在Asp.mvc中使用? public class Repository { //实例化EF容器:有弊端.一个线程里可能会创建多个DbContext //DbContext ...
- wampserver服务器无法启动(图标颜色不对)
1.服务器一直无法启动,图标颜色一直显示黄色,并且不可用. 2.解决办法: (1) C:\wamp\bin\apache\apache2.4.9\bin ->httpd.exe 找到该文件 ( ...
- SIMLock锁卡功能解析
一.锁卡背景介绍 锁卡即SIMLock,当手机开机启动或者插入SIM卡时,手机modem侧预置在NV项中的配置信息会与SIM卡中的信息做比对,检测是否匹配.若匹配,则SIM卡可以正常使用.若不匹配,则 ...
- HTML5 数据集属性dataset
有时候在HTML元素上绑定一些额外信息,特别是JS选取操作这些元素时特别有帮助.通常我们会使用getAttribute()和setAttribute()来读和写非标题属性的值.但为此付出的代价是文档将 ...
- [连载]《C#通讯(串口和网络)框架的设计与实现》- 13.中英文版本切换设计
目 录 第十三章 中英文版本切换设计... 2 13.1 不用自带的资源文件的理由... 2 13.2 配置文件... 2 13.3 语言 ...
- JAVA调用R
JAVA很适合开发应用系统,但是数学建模和计算能力非其所长,如果该系统需要进行大量的统计或者优化的计算,调用R是一种很好的方式.JAVA负责系统的构建,R用来做运算引擎,从而实现应用型和分析性相结合的 ...
- phpexcel导出数据表格
1.下载phpexcel(李昌辉) 2.在页面引入phpexcel的类文件,并且造该类的对象 include("../chajian/phpexcel/Classes/PHPExcel.ph ...