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";
}
}