1 MVP模式的三個角色
1 .Persenter-交互的中間人
主要作為View和model的橋梁,負責書寫一些業(yè)務邏輯和一些網絡通信請求通過model來拿到數(shù)據曹阔,返回給View顯示半开,這樣就可以完全的把View和model層分開
2.View-用戶界面
通常指的是Activity 和 fragment或者 某個view,他具有一個 Presenter的成員變量赃份。通常需要他去實現(xiàn)一個view的邏輯接口寂拆,將view的上一些操作轉給Presenter來實現(xiàn),最后Presenter將結果返回 戰(zhàn)士在view的上面抓韩。
3.Model-數(shù)據的存取
其實就是網路獲取數(shù)據或者本地化數(shù)據存儲的一個方式操作纠永,(可以簡單的理解為網絡請求,只要是數(shù)據的操作都放著里面沒毛糙怂)
2 下面來做一個簡單的獲取圖片鏈接來加載到Imageview上的操作
1定義一個 Presenter
public class MainPresenter {
//view的接口 角色
MainViewInterface mainViewInterfaceView;
//數(shù)據接口校色
TextModel textModel = new TextModel();
public MainPresenter(MainViewInterface mainViewInterface) {
mainViewInterfaceView = mainViewInterface;
}
;
public void getImageurl() {
textModel.getUrl();
mainViewInterfaceView.loadImageview(textModel.getUrl());
mainViewInterfaceView.toast();
}
}
該Presenter持有了MainViewInterface的引用和TextModel的引用尝江,MainViewInterface為主界面的邏輯接口 比如顯示數(shù)據 彈出toast等2 接口代碼如下:
public interface MainViewInterface {
//加載圖片
public void loadImageview(String url);
//線視提示
public void toast();
}
一些邏輯,你可以自行擴展 比如加載dialog 或者一些其他操作英上,都可以定義為接口可以自行返回數(shù)據或者不返回3.Activity實現(xiàn)過程
public class MainActivity extends Activity implements MainViewInterface {
private LinearLayout layout;
private ImageView imageView;
private Button button;
private Context mConetext;
MainPresenter mainPresenter ;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//實例化 Presentrer
mainPresenter = new MainPresenter(this);
mConetext = this;
imageView = findViewById(R.id.imga_pic);
button = findViewById(R.id.clearbu);
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
mainPresenter.getImageurl();
}
});
}
@Override
public void loadImageview(String url) {
Glide.with(mConetext).load(url).into(imageView);
}
@Override
public void toast() {
Toast.makeText(mConetext, "666", Toast.LENGTH_SHORT).show();
}
}
該Acticity實現(xiàn)了主界面的邏輯接口炭序,Presenter正好處理數(shù)據 而數(shù)據的展示剛好就交給Activity,獲取數(shù)據正好就交給Model層來進行獲取苍日,到此為止MVP模式基本成型
2 關于Activity和fragment的的生命周期問題
1 由于Presemnter經常要做一些網絡請求數(shù)據操作耗時操作惭聂,如果耗時操作在結束之前 activity被銷毀了,請求未返回導致Presenter一直持有Activity的對象 導致無法回收相恃,造成內存泄漏為了解決這個問題 可以采用弱引用辜纲。建立一個BasePresenter代碼如下:
public abstract class BasePresenter<T> {
//view 的 接口類型的 弱引用
protected Reference<T> mVieRef;
//建立 關聯(lián)
public void attachView(T view) {
mVieRef = new WeakReference<T>(view);
}
protected T getView() {
return mVieRef.get();
}
public boolean isViewAttached() {
return mVieRef != null && mVieRef.get() != null;
}
public void detachView() {
if (mVieRef != null) {
mVieRef.clear();
mVieRef = null;
}
}
}
定義自己的Presenter來繼承這ta,繼承的時候傳遞自己定義的ViewInterface接口 來實現(xiàn)綁定
2 建立MvpBaseActivity基類:
public abstract class MVPBaseActivity<V, T extends BasePresenter<V>> extends Activity {
protected T mPresenter;
@SuppressWarnings("unchecked")
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mPresenter = createPresenter();
mPresenter.attachView((V) this);
}
@Override
protected void onDestroy() {
super.onDestroy();
if (mPresenter.isViewAttached()) {
mPresenter.detachView();
}
}
protected abstract T createPresenter();
}
在destory生命周進行判斷 解除關聯(lián)豆茫,避免內存泄漏
實際應用
public class MainActivity extends MVPBaseActivity<MainViewInterface, MainPresenter> implements MainViewInterface {
private LinearLayout layout;
private ImageView imageView;
private Button button;
private Context mConetext;
MainPresenter mainPresenter = new MainPresenter(this);
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mConetext = this;
imageView = findViewById(R.id.imga_pic);
button = findViewById(R.id.clearbu);
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
mainPresenter.getImageurl();
}
});
}
@Override
protected MainPresenter createPresenter() {
return mainPresenter;
}
@Override
public void loadImageview(String url) {
Glide.with(mConetext).load(url).into(imageView);
}
@Override
public void toast() {
Toast.makeText(mConetext, "666", Toast.LENGTH_SHORT).show();
}
}
是不是非常簡單 沒有你想的那么復雜侨歉,記住一句話 解耦 分層,Presenter用來中轉數(shù)據 各司其職就好揩魂。補充一下Model的獲取數(shù)據代碼 幽邓,模擬一下網絡數(shù)據,請允許我偷個懶火脉。哈哈哈
public class TextModel {
//獲取網絡數(shù)據
public String getUrl() {
//模擬網絡請求
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
return "http://www.hao4k.com/data/attachment/forum/201705/27/154118c85uw88wwsmwajoj.jpg";
}
}