Android 简单封装一个MVP基类

对于MVP架构我想大家都不陌生吧,对于现在的主流框架非MVP + RxJava + Retrofit莫属了,GitHub上也有很多优秀的开源项目对其进行了很好的封装。有时候看一些优秀的开源项目是一种享受,但自己如果能从这里学到并实现一些自己想实现的需求我想那更是一种享受。这里很简单的去实现一个MVP的基类,其它扩展功能可以自行添加。

Model
public abstract class BaseModel {

}
Presenter
public abstract class BasePresenter<M,V> {

    public M mIModle;
    public V mIView;

    public void attachV(V v){
        this.mIView = v;
        if (mIModle == null) {
            mIModle = createModel();
        }
    }

    //当View被销毁掉时删除Presenter层对View层的引用
    public void detachV(){
        mIView = null;
    }

    public abstract M createModel();
}
View
public interface IBaseView {

}
@SuppressLint("Registered")
public abstract class BaseMvpActivity<P extends BasePresenter> extends AppCompatActivity implements IBaseView{

    public P mPresenter;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        if (mPresenter == null) {
            mPresenter = createPresenter();
        }

        mPresenter.attachV(this);
    }


    //当View被销毁掉时删除Presenter层对View层的引用
    @Override
    protected void onDestroy() {
        super.onDestroy();
        mPresenter.detachV();
    }

    public abstract P createPresenter();

}

这里Mode和View的基类里什么都没有,其实我们知道一般封装基类的目录是为了简化项目代码,提高开发效率。所以这里两个基类里面一般根据我们自己需求去实现。

Example
public interface ExampleDataSource {

    interface OnExampleFinishListener{
        void loading();
        void onSuccess();
        void onError();
    }

    void load(OnExampleFinishListener listener);
}

ExampleDataSource接口实现的是ExampleModel,而在它的内部接口OnExampleFinishListener是Model层的回调接口,它们真正实现是在Presenter层。对于获取到数据后(成功或失败)就是通过这个回调接口将数据传递Presenter层。

public class ExampleModel extends BaseModel implements ExampleDataSource {
    /**
     * model层主要用来进行数据存储操作,例如:数据读写和网络请求。
     * 这里进行下模拟耗时操作
     * @param listener
     */
    @Override
    public void load(final OnExampleFinishListener listener) {
        //加载中状态
        listener.loading();
        new Handler().postDelayed(new Runnable() {
            @Override
            public void run() {
                //获取成功后进行回调
                listener.onSuccess();
            }
        },2000);
    }
}

这里我们在Model层模拟下耗时操作(例如:网络请求),然后将得到的数据结果通过回调方法传递到Presenter层。

public interface ExampleContract extends IBaseView {

    interface ExamplePresent {
        void loadData();
    }

    interface ExampleView {
        void loadResult(String result);

        void showLoading();

        void loadFinish();
    }

}

这里我们将Presenter接口和View接口放在一个类中,这样可以使我们能够清晰的看到在Presenter层和View层中有哪些功能,方便我们以后维护。

public class ExamplePresenter extends BasePresenter<ExampleModel,ExampleContract.ExampleView> implements ExampleContract.ExamplePresent,ExampleDataSource.OnExampleFinishListener{

    @Override
    public void loadData() {
        mIModle.load(this);
    }

    @Override
    public void loading() {
        mIView.showLoading();
    }

    @Override
    public void onSuccess() {
        mIView.loadResult("成功");
        mIView.loadFinish();
    }

    @Override
    public void onError() {

    }

    @Override
    public ExampleModel createModel() {
        return new ExampleModel();
    }
}

从Presenter层我们可以看出,首先调用Model层的接口获取数据进行一些逻辑操作然后通过View层接口实现数据的展示。

public class MainActivity extends BaseMvpActivity<ExamplePresenter> implements ExampleContract.ExampleView {

    private TextView tvName;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        tvName = findViewById(R.id.tv_name);
        //加载数据
        mPresenter.loadData();
    }

    @Override
    public ExamplePresenter createPresenter() {
        return new ExamplePresenter();
    }

    //返会数据结果
    @Override
    public void loadResult(String result) {
        tvName.setText(result);
    }

    //加载过程
    @Override
    public void showLoading() {
        tvName.setText("正在加载,请稍等。。。");
    }

    //加载结束
    @Override
    public void loadFinish() {

    }
}

嗯,好了!一个很简单的MVP基类就实现了,对于使用轮子来说,我们更应该试着一点点自己去实现它,从简单到复杂。慢慢就会更加的了解它。贴上项目源码更参考。

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
好的,以下是一个简单MVP 框架,包含 Okhttp+Retrofit 网络封装,Base 基类的抽取以及 APPLocation 的代码: 1. 首先创建一个 BaseView 接口,定义一些公共的 UI 操作方法: ```java public interface BaseView { void showLoading(); void hideLoading(); void showError(String error); } ``` 2. 接着创建一个 BasePresenter 类,定义一些公共的 Presenter 操作方法: ```java public class BasePresenter<V extends BaseView> { private WeakReference<V> mViewRef; public void attachView(V view) { mViewRef = new WeakReference<>(view); } public void detachView() { if (mViewRef != null) { mViewRef.clear(); mViewRef = null; } } public boolean isViewAttached() { return mViewRef != null && mViewRef.get() != null; } public V getView() { return mViewRef.get(); } public void checkViewAttached() { if (!isViewAttached()) throw new RuntimeException("Please call attachView() before requesting data to the Presenter"); } } ``` 3. 创建一个 BaseActivity 类,作为所有 Activity 的基类,包含一些公共的操作: ```java public abstract class BaseActivity<P extends BasePresenter> extends AppCompatActivity implements BaseView { protected P mPresenter; @Override protected void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(getLayoutId()); mPresenter = createPresenter(); if (mPresenter != null) { mPresenter.attachView(this); } initView(); } @Override protected void onDestroy() { super.onDestroy(); if (mPresenter != null) { mPresenter.detachView(); } } protected abstract int getLayoutId(); protected abstract P createPresenter(); protected abstract void initView(); } ``` 4. 接着创建一个 BaseFragment 类,作为所有 Fragment 的基类,也包含一些公共的操作: ```java public abstract class BaseFragment<P extends BasePresenter> extends Fragment implements BaseView { protected P mPresenter; @Override public void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); mPresenter = createPresenter(); if (mPresenter != null) { mPresenter.attachView(this); } } @Override public void onDestroy() { super.onDestroy(); if (mPresenter != null) { mPresenter.detachView(); } } protected abstract P createPresenter(); } ``` 5. 创建一个 AppApplication 类,作为整个应用程序的入口,包含一些公共的配置信息和初始化操作: ```java public class AppApplication extends Application { private static AppApplication sInstance; @Override public void onCreate() { super.onCreate(); sInstance = this; // 初始化网络请求 RetrofitClient.getInstance().init(this); } public static AppApplication getInstance() { return sInstance; } } ``` 6. 创建一个 RetrofitClient 类,用于封装 Okhttp+Retrofit 网络请求: ```java public class RetrofitClient { private static final String TAG = "RetrofitClient"; private static final int DEFAULT_TIMEOUT = 30; private Retrofit mRetrofit = null; private OkHttpClient mOkHttpClient = null; private RetrofitClient() {} public static RetrofitClient getInstance() { return SingletonHolder.INSTANCE; } private static class SingletonHolder { private static final RetrofitClient INSTANCE = new RetrofitClient(); } public void init(Context context) { HttpLoggingInterceptor loggingInterceptor = new HttpLoggingInterceptor(); loggingInterceptor.setLevel(HttpLoggingInterceptor.Level.BODY); mOkHttpClient = new OkHttpClient.Builder() .connectTimeout(DEFAULT_TIMEOUT, TimeUnit.SECONDS) .readTimeout(DEFAULT_TIMEOUT, TimeUnit.SECONDS) .writeTimeout(DEFAULT_TIMEOUT, TimeUnit.SECONDS) .addInterceptor(loggingInterceptor) .addInterceptor(new TokenInterceptor()) .build(); mRetrofit = new Retrofit.Builder() .baseUrl(ApiService.BASE_URL) .client(mOkHttpClient) .addConverterFactory(GsonConverterFactory.create()) .addCallAdapterFactory(RxJava2CallAdapterFactory.create()) .build(); } public ApiService getApiService() { return mRetrofit.create(ApiService.class); } } ``` 7. 创建一个 ApiService 接口,定义所有的网络请求接口: ```java public interface ApiService { String BASE_URL = "https://www.example.com/"; @POST("login") Observable<BaseResponse<User>> login(@Query("username") String username, @Query("password") String password); } ``` 8. 最后,我们可以创建一个 LoginPresenter 类,来处理登录相关的业务逻辑: ```java public class LoginPresenter extends BasePresenter<LoginContract.View> implements LoginContract.Presenter { private ApiService mApiService; public LoginPresenter() { mApiService = RetrofitClient.getInstance().getApiService(); } @Override public void login(String username, String password) { checkViewAttached(); getView().showLoading(); mApiService.login(username, password) .subscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()) .subscribe(new Observer<BaseResponse<User>>() { @Override public void onSubscribe(Disposable d) { } @Override public void onNext(BaseResponse<User> userBaseResponse) { getView().hideLoading(); if (userBaseResponse.getCode() == 0) { getView().loginSuccess(userBaseResponse.getData()); } else { getView().showError(userBaseResponse.getMsg()); } } @Override public void onError(Throwable e) { getView().hideLoading(); getView().showError(e.getMessage()); } @Override public void onComplete() { } }); } } ``` 以上就是一个简单MVP 框架的实现,您可以根据自己的需求进行修改和扩展。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值