這篇文章是接著MVP架構(gòu)進(jìn)階(一)http://www.itdecent.cn/p/5a21942acb29寫(xiě)的,上篇沒(méi)有看,建議大家不要看,看懂了第一篇,有了一定的思路,在看這篇。這篇主要解決View層一對(duì)多個(gè)Presenter情況。
工程目錄圖

project.png
說(shuō)明:把上一篇的代碼放到了simple1中,這篇寫(xiě)的代碼在simple2中;
InjectPresenter:注入Presenter的注解;
具體思路:采用依賴注入的方式在v層注入p,怎么實(shí)現(xiàn),看代碼
InjectPresenter
//Target代表放在哪里使用 FIELD屬性 METHOD 方法 TYPE類
@Target(ElementType.FIELD)
//什么時(shí)候起作用,程序運(yùn)行起作用 RUNTIME運(yùn)行時(shí) CLASS編譯時(shí) SOURCE編程階段
@Retention(RetentionPolicy.RUNTIME)
public @interface InjectPresenter {
}
BaseMvpActivity
public abstract class BaseMvpActivity extends AppCompatActivity implements BaseView {
// private P mPresenter;
//一個(gè)View 里面肯定有多個(gè)Presenter情況,怎么處理,Dagger處理
private List<BasePresenter> mPresenters;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(getLayoutId());
mPresenters = new ArrayList<>();
//注入Presenter 通過(guò)反射(或者Dagger)
Field[] fields = this.getClass().getDeclaredFields();
for (Field field : fields) {
InjectPresenter injectPresenter = field.getAnnotation(InjectPresenter.class);
if (injectPresenter != null) {
//創(chuàng)建注入
Class<? extends BasePresenter> presenterClazz = null;
try {
//獲取這個(gè)類
presenterClazz = ( Class<? extends BasePresenter> ) field.getType();
} catch (Exception e) {
// 亂七八糟一些注入
throw new RuntimeException("not support inject presenter" + field.getType());
}
try {
//創(chuàng)建BasePresenter對(duì)象
BasePresenter basePresenter = presenterClazz.newInstance();
//attach
basePresenter.attach(this);
mPresenters.add(basePresenter);
field.setAccessible(true);
field.set(this, basePresenter);
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
}
}
initView();
initData();
}
protected abstract void initView();
protected abstract void initData();
public abstract int getLayoutId();
@Override
protected void onDestroy() {
super.onDestroy();
//解綁Presenter
for (BasePresenter presenter : mPresenters) {
presenter.detach();
}
}
利用了反射去注入Presenter,也可用Dagger2,Dagger2是在編譯期間,我這個(gè)運(yùn)行期間,建議大家使用Dagger2,但是這樣寫(xiě)也不是不可以。其效率肯定是編譯期間的注解優(yōu)于運(yùn)行期間的注解。
MainActivity
public class MainActivity extends BaseMvpActivity implements UserInfoContract.UserInfoView {
//多個(gè)Presenter怎么處理 dagger處理,自己寫(xiě)dagger處理 自己寫(xiě)個(gè)注入
//一個(gè)View 里面肯定有多個(gè)Presenter情況,怎么處理,Dagger處理
@InjectPresenter
private UserInfoPresenter mUserInfoPresenter;
private TextView mTextView;
@Override
public int getLayoutId() {
return R.layout.activity_main;
}
/**
* 初始化View
*/
@Override
protected void initView() {
mTextView = findViewById(R.id.tv);
}
/**
* 在這里去請(qǐng)求數(shù)據(jù)
*/
@Override
protected void initData() {
mUserInfoPresenter.getUserInfo("Steven");
}
/**
* 顯示一個(gè)加載的進(jìn)度條
*/
@Override
public void onLoading() {
}
/**
* 請(qǐng)求數(shù)據(jù)成功回調(diào)該方法
*
* @param user
*/
@Override
public void onSuccess(User user) {
mTextView.setText("Hello:" + user.getUserName());
}
/**
* 請(qǐng)求數(shù)據(jù)失敗回調(diào)該方法
*/
@Override
public void onError(int code, String errorMessage) {
}
大家可能會(huì)使用Dagger2去注入Presenter,這篇文章主要是在運(yùn)行期間去動(dòng)態(tài)注入Presenter,大家看下這個(gè)思路,體會(huì)下。哈,其實(shí)你會(huì)在很多源碼看到這種方式,運(yùn)行期間去動(dòng)態(tài)注入xxx。