與孤獨為伴铭腕,讓自己來一場涅槃
項目GitHub地址:https://github.com/ms-liu/ProjectFrameDemo
一券时、了解
在傳統(tǒng)的Android項目開發(fā)過程中,按照MVC模塊劃分的話黄选,往往會發(fā)現(xiàn)其實并不能很好的劃分出各自的職責(zé)出來蝇摸,Activity有時既需要扮演C的角色還需要扮演V的角色。這就導(dǎo)致Activity中代碼耦合現(xiàn)象嚴(yán)重办陷,尤其對于某些業(yè)務(wù)邏輯相對復(fù)雜的頁面貌夕,動不動就是上千行的代碼。
??可能對于這些頁面一代開發(fā)人員來說還能很愉快的接受民镜,但是當(dāng)新人來維護時啡专,這就會讓他很頭痛了;
??“他在寫些什么? (黑人問號 黑人問號)他應(yīng)該是個大神制圈,代碼我都看不懂们童,我要好好研究研究畔况!”
??在這種現(xiàn)象的基礎(chǔ)上,便有了MVP的開發(fā)模式:
- Model:數(shù)據(jù)層——業(yè)務(wù)邏輯和實體類
- View:視圖層——頁面展示
- Presenter:邏輯層——數(shù)據(jù)和視圖層交互
單單從目前的分析來看慧库,MVP和MVC并沒有什么大的區(qū)別跷跪,當(dāng)然這兩者本身本質(zhì)上的區(qū)別不大,都是用作解耦V和M齐板,只是將MVC套到Android開發(fā)過程中的時候吵瞻,讓Activity等一些組件,角色扮演不是那么清晰甘磨,所以導(dǎo)致了問題產(chǎn)生橡羞。
??而在MVP的模式中,將Activity這些組件完全當(dāng)成View層宽档,讓Activity職責(zé)單一化尉姨。Presenter負(fù)責(zé)數(shù)據(jù)處理,然后通過接口的形式達(dá)到與View交互的目的吗冤。讓View和Model不在有交集又厉。
??這篇文章不作MVP如何編寫的講解,如需了解可自行百度椎瘟,另外項目里面也包含了MVP 編寫代碼覆致。
二、提升與改進(重點與目的)
其實有過了解或者使用MVP的開發(fā)人員肺蔚,應(yīng)當(dāng)都能體會感受到煌妈,MVP確實能夠?qū)ctivity中代碼簡化。但是對于那些業(yè)務(wù)復(fù)雜界面宣羊,Presenter中代碼也是會急劇增多璧诵,并且有時也會將View層中代碼放到Presenter中,重新回到老狀態(tài)仇冯。
??另外不知道有沒有人碰到和我一樣的問題之宿,就是在使用MVP的過程中,有時在打開多個Activity頁面后苛坚,回退過程中比被,在某些使用Fragment的Activity中會報出NullPointException,這是因為在我們打開過多頁面時泼舱,由于內(nèi)存和生命周期管理導(dǎo)致Fragment被系統(tǒng)回收等缀,但是我們并沒有將這些告知Presenter,從而發(fā)生NullPointException娇昙。
??下面我們就一起來解決尺迂,這些問題。
(一)框架分析
??在該框架中,我們在V和P之間加上了一個Proxy或Controller代理類枪狂,我們將會盡量少的讓P和V直接進行交互接觸危喉,而是通過Proxy與V進行交互宋渔,在Proxy中預(yù)先處理部分邏輯州疾,從而達(dá)到減輕Presenter職責(zé)的目的,讓Presenter中代碼更加簡潔皇拣,更加專注于業(yè)務(wù)處理邏輯严蓖。
(二)Code
1、Model層
- 定義IModel數(shù)據(jù)接口
public interface IModel<T> {
void setModel(T t);
T getModel();
}
- 實現(xiàn)IModel接口
public class ImproveModelImpl implements IModel<ImproveInfoBean>{
private ImproveInfoBean mModel;
@Override
public void setModel(ImproveInfoBean improveInfoBean) {
this.mModel = improveInfoBean;
}
@Override
public ImproveInfoBean getModel() {
if (this.mModel == null){
this.mModel = new ImproveInfoBean();
}
return mModel;
}
}
2氧急、View層代碼
- 定義View生命周期監(jiān)聽接口
public interface OnViewStateListener {
void onCreate();
void onPause();
void onResume();
void onStop();
void onDestroy();
}
- 定義公共IView接口
public interface IView {
// 綁定對View生命周期監(jiān)聽
void bindListener(OnViewStateListener listener);
Context getContext();
// Toast提示
void showToast(String message);
//顯示加載對話框
void showLoadingDialog(String message);
// 隱藏加載對話框
void hideLoadingDialog();
}
- 編寫ViewDelegate
對 View操作的委托類颗胡,實現(xiàn)IView和OnViewStateListener接口中的方法
public class ViewDelegate implements IView,OnViewStateListener {
private Context mCtx;
private ProgressDialog mProgressDialog;
public ViewDelegate(Context context){
this.mCtx = context;
}
private List<OnViewStateListener> mOnViewStateListeners ;
@Override
public void bindListener(OnViewStateListener listener) {
//用數(shù)組管理每一個View的生命周期,避免一個頁面有多個監(jiān)聽
if (mOnViewStateListeners == null){
mOnViewStateListeners = new ArrayList<>();
mOnViewStateListeners.add(listener);
}else {
if (!mOnViewStateListeners.contains(listener)){
mOnViewStateListeners.add(listener);
}
}
}
@Override
public Context getContext() {
return mCtx;
}
@Override
public void showToast(String message) {
if (mCtx != null) {
Toast.makeText(mCtx, message, Toast.LENGTH_SHORT).show();
}
}
@Override
public void showLoadingDialog(String message) {
if (mCtx != null){
mProgressDialog = new ProgressDialog(mCtx);
}
}
@Override
public void hideLoadingDialog() {
if (mProgressDialog != null && mProgressDialog.isShowing()){
mProgressDialog.hide();
}
}
//------------------View生命周期管理--------------------------------------------------------
@Override
public void onCreate() {
if (checkListener()){
for (OnViewStateListener listener:
mOnViewStateListeners) {
listener.onCreate();
}
}
}
@Override
public void onPause() {
if (checkListener()){
for (OnViewStateListener listener:
mOnViewStateListeners) {
listener.onPause();
}
}
}
@Override
public void onResume() {
if (checkListener()){
for (OnViewStateListener listener:
mOnViewStateListeners) {
listener.onResume();
}
}
}
@Override
public void onStop() {
if (checkListener()){
for (OnViewStateListener listener:
mOnViewStateListeners) {
listener.onStop();
}
}
}
@Override
public void onDestroy() {
if (checkListener()){
for (OnViewStateListener listener:
mOnViewStateListeners) {
listener.onDestroy();
}
}
}
private boolean checkListener(){
return mOnViewStateListeners != null && !mOnViewStateListeners.isEmpty();
}
}
- 編寫B(tài)aseActivity吩坝,在里面實現(xiàn)頁面公共操作方法
public class BaseActivity extends AppCompatActivity implements IView{
private ViewDelegate mDelegate;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//可以考慮注解創(chuàng)建
mDelegate = new ViewDelegate(this);
}
@Override
public void bindListener(OnViewStateListener listener) {
mDelegate.bindListener(listener);
}
@Override
public Context getContext() {
return BaseActivity.this;
}
@Override
public void showToast(String message) {
mDelegate.showToast(message);
}
@Override
public void showLoadingDialog(String message) {
mDelegate.showLoadingDialog(message);
}
@Override
public void hideLoadingDialog() {
mDelegate.hideLoadingDialog();
}
}
3毒姨、Presenter層代碼
- IPresenter公共接口,因為要監(jiān)聽View生命周期钉寝,讓它繼承OnViewStateListener 接口
public interface IImprovePresenter<M,V extends IView> extends OnViewStateListener {
/**
* View綁定
* @param v
*/
void bindView(V v);
/**
* 數(shù)據(jù)加載
* @return
*/
M loadModel();
/**
* View解綁
*/
void detachView();
}
- 編寫B(tài)asePresenter弧呐,實現(xiàn)部分共方法
public abstract class BaseImprovePresenter<M extends IModel,V extends IView> implements IImprovePresenter<M,V> {
private List<String> mMethods;
public BaseImprovePresenter(){
this.mMethods = new ArrayList<>();
}
@Override
public void onCreate() {
}
@Override
public void onPause() {
}
@Override
public void onResume() {
}
@Override
public void onStop() {
}
@Override
public void onDestroy() {
}
//---------------------模擬需要在生命周期中處理的邏輯--------------------------------------
/**
* 添加View生命周期結(jié)束時,需要結(jié)束的方法名稱;
* 類似RxJava中添加addSubscription()
* @param methodName
*/
public void addHandleMethod(String methodName){
if (mMethods != null){
mMethods.add(methodName);
}
}
/**
* 清楚
* 類似RxJava中添加clearSubscription()
*/
public void clearMethod(){
if (mMethods != null && !mMethods.isEmpty()){
mMethods.clear();
}
}
}
- 編寫Presenter嵌纲,
public class ImprovePresenter extends BaseImprovePresenter<ImproveModelImpl,IImproveView> {
private IImproveView mView;
private ImproveModelImpl mModel;
@Override
public void bindView(IImproveView iImproveView) {
this.mView = iImproveView;
mModel = loadModel();
mView.showExplain(mModel.getModel().explain);
}
@Override
public ImproveModelImpl loadModel() {
addHandleMethod("異步請求數(shù)據(jù)方法");
return new ImproveModelImpl();
}
@Override
public void detachView() {
clearMethod();
}
public void setInfo(String username, String password) {
if (mModel != null){
mModel.getModel().setName(username);
mModel.getModel().setPassword(password);
}
}
public void getInfoBean(){
mView.showInfo("用戶名:"+mModel.getModel().getName()+"\r\n密碼:"+mModel.getModel().getPassword());
}
}
- 編寫PresenterProxy代理類
public class ImprovePresenterProxy implements IImprovePresenter<ImproveModelImpl,IImproveView>, View.OnClickListener {
private ImprovePresenter mPresenter;
private EditText etUserName;
private EditText etPassword;
private TextView componentShowInfo;
private TextView componentExplain;
private IImproveView mView;
public ImprovePresenterProxy(ImprovePresenter improvePresenter){
//判空
checkPresenter(improvePresenter);
//可以考慮依賴注入 方式
this.mPresenter = improvePresenter;
loadModel();
}
private void checkPresenter(ImprovePresenter improvePresenter) {
if (improvePresenter == null){
throw new NotBindPresenterException();
}
}
@Override
public ImproveModelImpl loadModel() {
return mPresenter.loadModel();
}
@Override
public void bindView(IImproveView iImproveView) {
checkView(iImproveView);
this.mView = iImproveView;
mPresenter.bindView(iImproveView);
}
private void checkView(IImproveView iImproveView) {
if (iImproveView == null){
throw new NotBindViewException();
}
}
@Override
public void detachView() {
mPresenter.detachView();
}
@Override
public void onCreate() {
//to do something
}
@Override
public void onPause() {
//to do something
}
@Override
public void onResume() {
//to do something
}
@Override
public void onStop() {
//to do something
}
@Override
public void onDestroy() {
detachView();
}
@Override
public void onClick(View view) {
int i = view.getId();
if (i == R.id.btn_save) {
if (etUserName != null && etPassword != null){
mView.showToast("保存成功");
mPresenter.setInfo(etUserName.getText().toString(),etPassword.getText().toString());
}
} else if (i == R.id.btn_get) {
mPresenter.getInfoBean();
} else {
}
}
public void setComponentName(EditText etUsername) {
this.etUserName = etUsername;
}
public void setComponentPassword(EditText etPassword) {
this.etPassword = etPassword;
}
public void setComponentShowInfo(TextView componentShowInfo) {
this.componentShowInfo = componentShowInfo;
}
public void setComponentExplain(TextView componentExplain) {
this.componentExplain = componentExplain;
}
}
至此俘枫,完成對MVP框架的完善和升級,完善后的MVP能夠更好的用于實際生產(chǎn)逮走,做到進一步的代碼解耦的目的鸠蚪。并且由于加入了對View生命周期的管理,也很好的解決NullPointException問題师溅。
實際代碼請下載或者Frok項目茅信。
項目GitHub地址:https://github.com/ms-liu/ProjectFrameDemo
歡迎大家給出中肯的建議和提高意見,大家一起學(xué)習(xí)進步墓臭。