第一:最簡單的MVP
看下項目結構

Module層
package com.example.testmvp.module;
import java.util.List;
/**
*
*/
public interface IShenModule {
void loadShenData(onLoadListenner onLoadListenner);
//module通過接口返回數(shù)據(jù),而不是通過return 返回數(shù)據(jù)
interface onLoadListenner{
void complete(List<String> shens);
}
}
package com.example.testmvp.module;
import java.util.ArrayList;
import java.util.List;
/**
* @author writing
* @time 2019/11/7 21:42
* @note
*/
public class ShenModule implements IShenModule {
@Override
public void loadShenData(onLoadListenner onLoadListenner) {
//提供數(shù)據(jù),實際開發(fā)中可能從網(wǎng)絡獲取
List<String> stringList = new ArrayList<>();
stringList.add("戰(zhàn)神呂布");
stringList.add("斗神張飛");
stringList.add("勇神孫策");
stringList.add("將神關羽");
stringList.add("槍神趙云");
stringList.add("騎神馬超");
stringList.add("戟神太史慈");
stringList.add("工神黃月英");
stringList.add("算神諸葛亮");
stringList.add("水神周泰");
stringList.add("火神周瑜");
stringList.add("箭神黃忠");
onLoadListenner.complete(stringList);
}
}
view層
package com.example.testmvp.view;
import java.util.List;
/**
*
* View接口,UI邏輯都定義在這里
*/
public interface IShenlView {
//顯示諸位大神
void showShenView(List<String> shens);
//也可以有其它邏輯,比如加載進度條等
}
presenter層
package com.example.testmvp.presenter;
import com.example.testmvp.module.IShenModule;
import com.example.testmvp.module.ShenModule;
import com.example.testmvp.view.IShenlView;
import java.util.List;
/**
* @author writing
* @time 2019/11/7 21:43
* @note
*/
public class ShenPresenter {
//持有View
private IShenlView iShenlView;
//持有module
private IShenModule iShenModule = new ShenModule();
public ShenPresenter(IShenlView iShenlView ) {
this.iShenlView = iShenlView;
}
//執(zhí)行UI邏輯
public void fetch(){
if (iShenlView!=null&&iShenModule!=null){
iShenModule.loadShenData(new IShenModule.onLoadListenner() {
@Override
public void complete(List<String> shens) {
iShenlView.showShenView(shens);
}
});
}
}
}
MainActivity
package com.example.testmvp;
import androidx.appcompat.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.ListView;
import com.example.testmvp.presenter.ShenPresenter;
import com.example.testmvp.view.IShenlView;
import java.util.ArrayList;
import java.util.List;
/**
* 1,MVP把UI邏輯抽象成VIEW接口,把業(yè)務邏輯抽象成Presenter接口。
*
*
*/
public class MainActivity extends AppCompatActivity implements IShenlView {
private ListView listView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
listView = findViewById(R.id.listview);
ShenPresenter presenter = new ShenPresenter(this);
presenter.fetch();
}
@Override
public void showShenView(List<String> shens) {
MyAdapter adapter = new MyAdapter(this,shens);
listView.setAdapter(adapter);
}
}
MyAdapter
package com.example.testmvp;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;
import java.util.List;
/**
* @author writing
* @time 2019/11/7 21:14
* @note
*/
public class MyAdapter extends BaseAdapter {
private Context context;
private List<String> stringList;
public MyAdapter(Context context, List<String> stringList) {
this.context = context;
this.stringList = stringList;
}
@Override
public int getCount() {
return stringList.size();
}
@Override
public Object getItem(int i) {
return stringList.get(i);
}
@Override
public long getItemId(int i) {
return i;
}
@Override
public View getView(int i, View view, ViewGroup viewGroup) {
View v=null;
if(view== null){
v= LayoutInflater.from(context).inflate(R.layout.listitem,null);
}else{
v=view;
}
TextView textView = v.findViewById(R.id.textView);
textView.setText(stringList.get(i));
return v;
}
}
布局就是一個ListView就不貼出來了
看下運行效果

第二 使用弱引用,把Presenter層與View層的聯(lián)系斷開。
以前我們在Activity中發(fā)起網(wǎng)絡請求,需要開辟一個子線程。如果子線程耗時任務太長,就會造成內存泄露。比如一個子線程,請求網(wǎng)絡花費了很多時間,這個時候我們點回退鍵,本要銷毀的Activity就會無法銷毀,造成了內存泄露。我們采用MVP就會徹底根除點Activity內存泄露的問題,我們可以把線程處理問題都移動到其它層去。Activity再也不用考慮數(shù)據(jù)加載等問題,只需要考慮UI邏輯。
我們現(xiàn)在修改Presenter層
package com.example.testmvp.presenter;
import com.example.testmvp.module.IShenModule;
import com.example.testmvp.module.ShenModule;
import com.example.testmvp.view.IShenlView;
import java.lang.ref.WeakReference;
import java.util.List;
/**
* @author writing
* @time 2019/11/7 21:43
* @note
*/
public class ShenPresenter<T extends IShenlView> {
/**
* 使用弱引用,保證內存不足的時候或者發(fā)生異常的時候會釋放掉這個引用,所以這里我們換一種寫法。
* 使用弱引用在View層就不會再有超時操作。徹底根除Activity的內存泄露。
*/
//持有View
// private IShenlView iShenlView;
WeakReference<T> iShenlView;
//持有module
private IShenModule iShenModule = new ShenModule();
public ShenPresenter(T view) {
iShenlView = new WeakReference<>(view);
}
//執(zhí)行UI邏輯
public void fetch() {
if (iShenlView.get() != null && iShenModule != null) {
iShenModule.loadShenData(new IShenModule.onLoadListenner() {
@Override
public void complete(List<String> shens) {
iShenlView.get().showShenView(shens);
}
});
}
}
}
其它代碼不變,效果依舊
第三android系統(tǒng)解決方案
聯(lián)想下,我們以前使用Activity,里面有很多fragment,當我們自己退出Activity的時候,從來不需要自己寫代碼釋放fragment。我們這里也可以使用生命周期管理presenter。聯(lián)想fragment的生命周期onAttach onDetach
我們修改Presenter層
package com.example.testmvp.presenter;
import com.example.testmvp.module.IShenModule;
import com.example.testmvp.module.ShenModule;
import com.example.testmvp.view.IShenlView;
import java.lang.ref.WeakReference;
import java.util.List;
/**
* @author writing
* @time 2019/11/7 21:43
* @note
*/
public class ShenPresenter<T extends IShenlView> {
/**
* 使用弱引用,保證內存不足的時候或者發(fā)生異常的時候會釋放掉這個引用,所以這里我們換一種寫法。
* 使用弱引用在View層就不會再有超時操作。徹底根除Activity的內存泄露。
*/
//持有View
WeakReference<T> iShenlView;
//持有module
private IShenModule iShenModule = new ShenModule();
//不需要在構造方法中寫,在attach方法中寫
// public ShenPresenter(T view) {
// iShenlView = new WeakReference<>(view);
//
// }
//執(zhí)行UI邏輯
public void fetch() {
if (iShenlView.get() != null && iShenModule != null) {
iShenModule.loadShenData(new IShenModule.onLoadListenner() {
@Override
public void complete(List<String> shens) {
iShenlView.get().showShenView(shens);
}
});
}
}
//不需要通過構造方法綁定
public void attachView(T view) {
iShenlView = new WeakReference<>(view);
}
public void detachView() {
if (iShenlView != null) {
//清除弱引用的數(shù)據(jù)
iShenlView.clear();
//弱引用置為空
iShenlView = null;
}
}
}
然后我們修改MainActivity
package com.example.testmvp;
import androidx.appcompat.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.ListView;
import com.example.testmvp.presenter.ShenPresenter;
import com.example.testmvp.view.IShenlView;
import java.util.ArrayList;
import java.util.List;
/**
* 1,MVP把UI邏輯抽象成VIEW接口,把業(yè)務邏輯抽象成Presenter接口。
*/
public class MainActivity extends AppCompatActivity implements IShenlView {
private ListView listView;
private ShenPresenter presenter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
listView = findViewById(R.id.listview);
// ShenPresenter presenter = new ShenPresenter(this);
presenter = new ShenPresenter();
presenter.attachView(this);
presenter.fetch();
}
//解綁
@Override
protected void onDestroy() {
super.onDestroy();
presenter.detachView();
}
@Override
public void showShenView(List<String> shens) {
MyAdapter adapter = new MyAdapter(this, shens);
listView.setAdapter(adapter);
}
}
第四引入父類
這樣寫肯定也不行,因為將來我們每建立一個表示層和activity都要這么寫。我們的解決辦法就是給表示層建立一個父類,進行統(tǒng)一管理
我們先看下修改后的項目結構

presenter層
package com.example.testmvp.presenter;
import com.example.testmvp.view.IShenView;
import java.lang.ref.WeakReference;
/**
* @author writing
* @time 2019/11/8 16:52
* @note
*/
public class BasePresenter<T extends IShenView> {
WeakReference<T> iShenlView;
//不需要通過構造方法綁定
public void attachView(T view) {
iShenlView = new WeakReference<>(view);
}
public void detachView() {
if (iShenlView != null) {
//清除弱引用的數(shù)據(jù)
iShenlView.clear();
//弱引用置為空
iShenlView = null;
}
}
}
package com.example.testmvp.presenter;
import com.example.testmvp.module.IShenModule;
import com.example.testmvp.module.ShenModule;
import com.example.testmvp.view.IShenView;
import java.util.List;
/**
* @author writing
* @time 2019/11/7 21:43
* @note
*/
public class ShenPresenter<T extends IShenView> extends BasePresenter<T>{
//將來我自己寫Presenter的時候只需這個數(shù)據(jù)來源就可以
private IShenModule iShenModule = new ShenModule();
//執(zhí)行UI邏輯
public void fetch() {
if (iShenlView.get() != null && iShenModule != null) {
iShenModule.loadShenData(new IShenModule.onLoadListenner() {
@Override
public void complete(List<String> shens) {
iShenlView.get().showShenView(shens);
}
});
}
}
}
View層和MainActivity
package com.example.testmvp.view;
import java.util.List;
/**
*
* View接口,UI邏輯都定義在這里
*/
public interface IShenView {
//顯示諸位大神
void showShenView(List<String> shens);
//也可以有其它邏輯,比如加載進度條等
}
package com.example.testmvp.view;
import android.os.Bundle;
import androidx.annotation.Nullable;
import androidx.appcompat.app.AppCompatActivity;
import com.example.testmvp.presenter.BasePresenter;
/**
* @author writing
* @time 2019/11/8 16:59
* @note
*/
public abstract class BaseActivity<T extends BasePresenter,V extends IShenView> extends AppCompatActivity {
//持有表示層
public T presenter;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
presenter = createPresenter();
presenter.attachView((V) this);
}
//繼承BaseActivity的Activity實現(xiàn)這個賦值動作
public abstract T createPresenter();
@Override
protected void onDestroy() {
super.onDestroy();
presenter.detachView();
}
}
package com.example.testmvp;
import android.os.Bundle;
import android.util.Log;
import android.widget.ListView;
import com.example.testmvp.presenter.ShenPresenter;
import com.example.testmvp.view.BaseActivity;
import com.example.testmvp.view.IShenView;
import java.util.List;
/**
* 1,MVP把UI邏輯抽象成VIEW接口,把業(yè)務邏輯抽象成Presenter接口。
*/
public class MainActivity extends BaseActivity<ShenPresenter<IShenView>, IShenView> implements IShenView {
private ListView listView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
listView = findViewById(R.id.listview);
Log.i("zhang_xin","后執(zhí)行");
presenter.fetch();
}
/**
* 選擇表示層
*/
@Override
public ShenPresenter<IShenView> createPresenter() {
Log.i("zhang_xin","先執(zhí)行");
return new ShenPresenter<>();
}
/**
* UI邏輯
*/
@Override
public void showShenView(List<String> shens) {
MyAdapter adapter = new MyAdapter(this, shens);
listView.setAdapter(adapter);
}
}
Module層不變
第五 一行代碼切換框架
我們考慮下這個場景,我們進行版本的迭代,我們上一個版本訪問網(wǎng)絡用的是retrofit,這次我們換成一個新的網(wǎng)絡訪問框架。我們現(xiàn)在就講解下一行代碼切換框架.
我們可以增加一個隔離層,當我們的app需要網(wǎng)絡操作的時候,由他幫我們選擇框架。