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客户端的开发.以下 ...
随机推荐
- 【译】Unity3D Shader 新手教程(4/6) —— 卡通shader(入门版)
本文为翻译,附上原文链接. 转载请注明出处--polobymulberry-博客园. 暗黑系 动机 如果你满足以下条件,我建议你阅读这篇教程: 你想了解更多有关表面着色器的细节知识. 你想实现一个入门 ...
- ASP.NET MVC5+EF6+EasyUI 后台管理系统(47)-工作流设计-补充
系列目录 补充一下,有人要表单的代码,这个用代码生成器生成表Flow_Form表的Index代码就可以 加上几个按钮就可以了 <div class="mvctool"> ...
- 难道.NET Core到R2连中文编码都不支持吗?
今天写了一个简单的.NET Core RC2控制台程序,发现中文显示一直是乱码.查看操作系统设置,没有问题:查看源文件编码,也没有问题:甚至查看了Console字符编码相关的注册表,依然没有发现问题. ...
- 【基于WPF+OneNote+Oracle的中文图片识别系统阶段总结】之篇二:基于OneNote难点突破和批量识别
篇一:WPF常用知识以及本项目设计总结:http://www.cnblogs.com/baiboy/p/wpf.html 篇二:基于OneNote难点突破和批量识别:http://www.cnblog ...
- 年终巨献 史上最全 ——LINQ to SQL语句
LINQ to SQL语句(1)之Where 适用场景:实现过滤,查询等功能. 说明:与SQL命令中的Where作用相似,都是起到范围限定也就是过滤作用的,而判断条件就是它后面所接的子句.Where操 ...
- <精通JavaScript>---阅读笔记01
下面是阅读精通JavaScript书做的相关笔记. JS中的函数重载 函数重载必须依赖两件事情:判断传入参数数量的能力和判断传入参数类型的能力,在js中每个函数都带有一个仅在这个函数范围内作用的变量, ...
- QT 删除文件指定目录
bool deleteDir(const QString &dirName) { QDir directory(dirName); if (!directory.exists()) { ret ...
- asp.net webform 自定义分页控件
做web开发一直用到分页控件,自己也动手实现了个,使用用户自定义控件. 翻页后数据加载使用委托,将具体实现放在在使用分页控件的页面进行注册. 有图有真相,给个直观的认识: 自定义分页控件前台代码: & ...
- js验证输入的是否是数字,小数保留几位小数
1.验证方法 validationNumber(e, num) e代表标签对象,num代表保留小数位数 function validationNumber(e, num) { -]+\.?[-]*$ ...
- String类
字符串的功能 A:判断功能 boolean equals(Object obj)//比较对象 boolean eq ...