MVP设计架构+生命周期封装防止内存泄漏

1 MVP模式的三个角色

1 .Persenter-交互的中间人

主要作为View和model的桥梁,负责书写一些业务逻辑和一些网络通信请求通过model来拿到数据,返回给View显示,这样就可以完全的把View和model层分开

2.View-用户界面

通常指的是Activity 和 fragment或者 某个view,他具有一个 Presenter的成员变量。通常需要他去实现一个view的逻辑接口,将view的上一些操作转给Presenter来实现,最后Presenter将结果返回 战士在view的上面。

3.Model-数据的存取

其实就是网路获取数据或者本地化数据存储的一个方式操作,(可以简单的理解为网络请求,只要是数据的操作都放着里面没毛病)

2 下面来做一个简单的获取图片链接来加载到Imageview上的操作

1定义一个 Presenter

public class MainPresenter {

   //view的接口 角色
MainViewInterface mainViewInterfaceView;
  //数据接口校色
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为主界面的逻辑接口 比如显示数据 弹出toast等

2 接口代码如下:
public interface MainViewInterface {

//加载图片
public void loadImageview(String url);

//线视提示
public void toast();

}
一些逻辑,你可以自行扩展 比如加载dialog 或者一些其他操作,都可以定义为接口可以自行返回数据或者不返回

3.Activity实现过程

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实现了主界面的逻辑接口,Presenter正好处理数据 而数据的展示刚好就交给Activity,获取数据正好就交给Model层来进行获取,到此为止MVP模式基本成型

2 关于Activity和fragment的的生命周期问题

1 由于Presemnter经常要做一些网络请求数据操作耗时操作,如果耗时操作在结束之前 activity被销毁了,请求未返回导致Presenter一直持有Activity的对象 导致无法回收,造成内存泄漏为了解决这个问题 可以采用弱引用。建立一个BasePresenter代码如下:
public abstract class BasePresenter<T> {

//view 的 接口类型的 弱引用
protected Reference<T> mVieRef;

//建立 关联
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接口 来实现绑定

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生命周进行判断 解除关联,避免内存泄漏

实际应用
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用来中转数据 各司其职就好。补充一下Model的获取数据代码 ,模拟一下网络数据,请允许我偷个懒。哈哈哈
public class TextModel {

//获取网络数据
public String getUrl() {
    //模拟网络请求
    try {
        Thread.sleep(2000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
    return "http://www.hao4k.com/data/attachment/forum/201705/27/154118c85uw88wwsmwajoj.jpg";
}

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容