Android開發(fā):淺談MVP模式應(yīng)用與內(nèi)存泄漏問題解決
最近博主開始在項目中實踐MVP模式,卻意外發(fā)現(xiàn)內(nèi)存泄漏比較嚴(yán)重,但卻很少人談到這個問題,促使了本文的發(fā)布,本文假設(shè)讀者已了解MVP架構(gòu)。
MVP簡介
M-Modle,數(shù)據(jù),邏輯操作層,數(shù)據(jù)獲取,數(shù)據(jù)持久化保存。比如網(wǎng)絡(luò)操作,數(shù)據(jù)庫操作
V-View,界面展示層,Android中的具體體現(xiàn)為Activity,F(xiàn)ragment
P-Presenter,中介者,連接Modle,View層,同時持有modle引用和view接口引用 
示例代碼
Modle層操作
public class TestModle implements IModle{
private CallbackListener callback;
public TestModle(CallbackListener callback) {
this.callback = callback;
}
public interface CallbackListener {
void onGetData(String data);
}
public void getData() {
new Thread() {
public void run() {
callback.onGetData("返回的數(shù)據(jù)");
}
}.start();
}
}
View層
// 抽象的view層
public interface TestViewInterf extends IView {
void onGetData(String data);
}
// 具體的View層
public class MainActivity extends Activity implements TestViewInterf{
private TestPresenter mTestPresenter;
@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// view層將獲取數(shù)據(jù)的任務(wù)委派給中介者presenter,并傳入自身實例對象,實現(xiàn)TestViewInterf接口
mTestPresenter = new TestPresenter(this);
mTestPresenter.getData();
}
@Override
public void onGetData(String data) {
// View層只做數(shù)據(jù)展示
showToast(data);
}
private void showToast(String toast) {
Toast.makeText(this, toast, Toast.LENGTH_LONG).show();
}
}
Presenter中介者
public class TestPresenter implements IPresenter{
IModle modle;
IView view;
public TestPresenter(IView view) {
this.view = view;
}
public void getData() {
// 獲取數(shù)據(jù)的操作實際在Modle層執(zhí)行
modle = new TestModle(new CallbackListener() {
public void onGetData(String data) {
if (view != null) {
view.onGetData(data);
}
}
});
modle.getData();
}
}
根據(jù)OOP思想,Java應(yīng)面向接口編程,這樣才能給符合OCP原則。上述示例代碼省略了更加抽象的接口IModle,IView,IPresenter,并且實際MVP實踐中通常會引入泛型使其更具擴展性。
Google已提供了相關(guān)示例代碼,并在MVP中增加了一個約束者:Contract,它的作用是定義各個模塊的MVP接口。
google MVP sample code:https://github.com/googlesamples/android-architecture
內(nèi)存泄露問題
由上可見,Presenter中持有View接口對象,這個接口對象實際為MainActivity.this,Modle中也同時擁有Presenter對象實例,當(dāng)MainActivity要銷毀時,Presenter中有Modle在獲取數(shù)據(jù),那么問題來了,這個Activity還能正常銷毀嗎?
答案是不能!
當(dāng)Modle在獲取數(shù)據(jù)時,不做處理,它就一直持有Presenter對象,而Presenter對象又持有Activity對象,這條GC鏈不剪斷,Activity就無法被完整回收。
換句話說:Presenter不銷毀,Activity就無法正常被回收。
解決MVP的內(nèi)存泄露
Presenter在Activity的onDestroy方法回調(diào)時執(zhí)行資源釋放操作,或者在Presenter引用View對象時使用更加容易回收的軟引用,弱應(yīng)用。
比如示例代碼:
Activity
@Override
public void onDestroy() {
super.onDestroy();
mPresenter.destroy();
}
Presenter
public void destroy() {
view = null;
if(modle != null) {
modle.cancleTasks();
}
}
Modle
public void cancleTasks() {
// TODO 終止線程池ThreadPool.shutDown(),AsyncTask.cancle(),或者調(diào)用框架的取消任務(wù)api
}
個人總結(jié)
因為面向MVP接口編程,可適應(yīng)需求變更,所以MVP適用于比較大的項目;因為其簡化了Activity和Fragmnt的職責(zé),可大大減少View層的代碼量,比起MVC中Activity,F(xiàn)ragment動不動上千行的代碼量,簡直優(yōu)雅!
做完以上操作,由于MVP引起的內(nèi)存泄露就差不多解決了,以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
android利用websocket協(xié)議與服務(wù)器通信
這篇文章主要為大家詳細介紹了android利用websocket協(xié)議與服務(wù)器通信,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-03-03
Android四大組件之廣播BroadcastReceiver詳解
Android開發(fā)的四大組件分別是:活動(activity),用于表現(xiàn)功能;服務(wù)(service),后臺運行服務(wù),不提供界面呈現(xiàn);廣播接受者(Broadcast Receive),勇于接收廣播;內(nèi)容提供者(Content Provider),支持多個應(yīng)用中存儲和讀取數(shù)據(jù),相當(dāng)于數(shù)據(jù)庫,本篇著重介紹廣播組件2021-11-11
Android實現(xiàn)透明度可變的標(biāo)題欄效果
這篇文章主要介紹了Android實現(xiàn)透明度可變的標(biāo)題欄效果的相關(guān)資料,具有一定的參考價值,需要的朋友可以參考下2016-02-02
Android實現(xiàn)帶動畫效果的可點擊展開TextView
這篇文章主要為大家詳細介紹了Android實現(xiàn)帶動畫效果的可點擊展開TextView,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-07-07
Android實現(xiàn)滑動折疊Header全流程詳解
這篇文章主要介紹了Android實現(xiàn)滑動折疊Header,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧2022-11-11
Android 組合控件實現(xiàn)布局的復(fù)用的方法
本篇文章主要介紹了Android 組合控件實現(xiàn)布局的復(fù)用的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-08-08
Android在listview添加checkbox實現(xiàn)原理與代碼
Android在listview添加checkbox如何實現(xiàn)一直都是新手朋友們的頭疼問題,接下來為您詳細介紹實現(xiàn)方法,感興趣的朋友可以了解下2013-01-01

