开发文章

Android的MVP模式应用与内存泄漏

MVP简介

M-Modle,数据,逻辑操作层,数据获取,数据持久化保存。比如网络操作,数据库操作
V-View,界面展示层,Android中的具体体现为Activity,Fragment
P-Presenter,中介者,连接Modle,View层,同时持有modle引用和view接口引用

MVP简介.png

上图摘自阮一峰大神博客:MVC,MVP 和 MVVM 的图示

注:有别于MVC,Activity,Fragment通常被用作Controller和View使用,加重了它的职责。在MVP中,Activity,Fragment仅用做View层展示

示例代码

本文原创作者:xiong_it,链接:http://blog.csdn.net/xiong_it

Modle层操作

复制内容到剪贴板
  1. public class TestModle implements IModle{  
  2.     private CallbackListener callback;  
  3.   
  4.     public TestModle(CallbackListener callback) {  
  5.         this.callback = callback;  
  6.     }  
  7.     public interface CallbackListener {  
  8.         void onGetData(String data);  
  9.     }  
  10.     public void getData() {  
  11.         new Thread() {  
  12.             public void run() {  
  13.                 callback.onGetData("返回的数据");  
  14.             }  
  15.         }.start();  
  16.     }  
  17. }  

View层

复制内容到剪贴板
  1. // 抽象的view层  
  2. public interface TestViewInterf extends IView {  
  3.     void onGetData(String data);  
  4. }  
  5.   
  6. // 具体的View层  
  7. public class MainActivity extends Activity implements TestViewInterf{  
  8.     private TestPresenter mTestPresenter;  
  9.   
  10.     @Override  
  11.     public void onCreate(@Nullable Bundle savedInstanceState) {  
  12.         super.onCreate(savedInstanceState);  
  13.   
  14.         // view层将获取数据的任务委派给中介者presenter,并传入自身实例对象,实现TestViewInterf接口  
  15.         mTestPresenter = new TestPresenter(this);  
  16.         mTestPresenter.getData();  
  17.     }  
  18.   
  19.     @Override  
  20.     public void onGetData(String data) {  
  21.         // View层只做数据展示  
  22.         showToast(data);  
  23.     }  
  24.   
  25.     private void showToast(String toast) {  
  26.         Toast.makeText(this, toast, Toast.LENGTH_LONG).show();  
  27.     }  
  28. }  

Presenter中介者

复制内容到剪贴板
  1. public class TestPresenter implements IPresenter{  
  2.     IModle modle;  
  3.     IView view;  
  4.     public TestPresenter(IView view) {  
  5.         this.view = view;  
  6.     }  
  7.   
  8.     public void getData() {  
  9.         // 获取数据的操作实际在Modle层执行  
  10.         modle = new TestModle(new CallbackListener() {  
  11.             public void onGetData(String data) {  
  12.                 if (view != null) {  
  13.                     view.onGetData(data);  
  14.                 }  
  15.             }  
  16.         });  
  17.         modle.getData();  
  18.     }  
  19. }  

根据OOP思想,Java代码最好面向接口,抽象编程,这样才能给符合OCP原则。上述示例代码省略了更加抽象的接口IModle,IView,IPresenter,并且实际MVP实践中通常会引入泛型使其更具扩展性。

Google已提供了相关示例代码,并在MVP中增加了一个约束者:Contract,它的作用是定义各个模块的MVP接口。
google MVP sample code:https://github.com/googlesamples/android-architecture

内存泄露问题

由上可见,Presenter中持有View接口对象,这个接口对象实际为MainActivity.this,Modle中也同时拥有Presenter对象实例,当MainActivity要销毁时,Presenter中有Modle在获取数据,那么问题来了,这个Activity还能正常销毁吗?
答案是不能!
当Modle在获取数据时,不做处理,它就一直持有Presenter对象,而Presenter对象又持有Activity对象,这条GC链不剪断,Activity就无法被完整回收。
换句话说:Presenter不销毁,Activity就无法正常被回收。

解决MVP的内存泄露

Presenter在Activity的onDestroy方法回调时执行资源释放操作,或者在Presenter引用View对象时使用更加容易回收的软引用,弱应用。
比如示例代码:
Activity

复制内容到剪贴板
  1. @Override  
  2.     public void onDestroy() {  
  3.         super.onDestroy();  
  4.         mPresenter.destroy();  
  5.         mPresenter = null;  
  6.     }  

Presenter

复制内容到剪贴板
  1. public void destroy() {  
  2.     view = null;  
  3.     if(modle != null) {  
  4.         modle.cancleTasks();  
  5.         modle = null;  
  6.     }  
  7. }  

Modle

复制内容到剪贴板
  1. public void cancleTasks() {  
  2.     // TODO 终止线程池ThreadPool.shutDown(),AsyncTask.cancle(),或者调用框架的取消任务api  
  3. }  

个人总结

因为面向MVP接口编程,可适应需求变更,所以MVP适用于比较大的项目;因为其简化了Activity和Fragmnt的职责,可大大减少View层的代码量,比起MVC中Activity,Fragment动不动上千行的代码量,简直优雅!

做完以上操作,由于MVP引起的内存泄露就差不多解决了,祝大家撸码愉快!欢迎留言区交流指正。

感谢 xiong_it 支持 磐实编程网 原文地址:
blog.csdn.net/xiong_it/article/details/52610729

文章信息

发布时间:2016-11-25

作者:xiong_it

发布者:aquwcw

浏览次数: