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客户端的开发.以下 ...
随机推荐
- 断电不断网——Linux的screen
title: 断电不断网--Linux的screen author:青南 date: 2015-01-01 20:20:23 categories: [Linux] tags: [linux,scre ...
- iOS开发之Alamofire源码解析前奏--NSURLSession全家桶
今天博客的主题不是Alamofire, 而是iOS网络编程中经常使用的NSURLSession.如果你想看权威的NSURLSession的东西,那么就得去苹果官方的开发中心去看了,虽然是英文的,但是结 ...
- Android之图片加载框架Fresco基本使用(二)
PS:最近看到很多人都开始写年终总结了,时间过得飞快,又到年底了,又老了一岁. 学习内容: 1.进度条 2.缩放 3.ControllerBuilder,ControllerListener,Post ...
- 深入学习jQuery特性操作
× 目录 [1]获取特性 [2]设置特性 [3]删除特性 前面的话 每个元素都有一个或者多个特性,这些特性的用途就是给出相应元素或者其内容的附加信息.操作特性的DOM方法主要有3个:getAttrib ...
- 如果你也会C#,那不妨了解下F#(2):数值运算和流程控制语法
本文链接:http://www.cnblogs.com/hjklin/p/fs-for-cs-dev-2.html 一些废话 一门语言火不火,与语言本身并没太大关系,主要看语言的推广. 推广得好,用的 ...
- Rafy 框架-发布网页版用户手册
前段时间把 Rafy 的用户手册由 CHM 格式转换为了网页格式,而且发布到了 github.io 上,即方便文档的实时更新,也方便大家查看. Rafy 用户手册网页版地址: http://zgynh ...
- cin.ignore()函数的用法
cin.ignore(a,ch)方法是从输入流(cin)中提取字符,提取的字符被忽略(ignore),不被使用.每抛弃一个字符,它都要计数和比较字符:如果计数值达到a或者被抛弃的字符是ch,则cin. ...
- mongodb在java中的查询
mongodb 根据_id 查询记录: public Price queryPriceById(String id) throws Exception { return mongoTemplate.f ...
- js Form.elements[i]的使用实例
function pdf(){ //一个html里面可能存在多个form,所以document.form[0]指的是第一个form,document.form[1]返回就是第二个form,如果没 ...
- java面试题——集合框架
先来看一下集合框架关系图 Collection FrameWork 如下: Collection ├List │├LinkedList │├ArrayList │└Vector │ └Stack └S ...