Android从零开始搭建MVVM架构(6) ———— 使用玩Android API带你搭建MVVM框架(初级篇)

在经历了半个月的AAC组件的学习,终于来到了最后一步。但愿本文可以帮助到你。本demo架构RxJava + Retrofit + MVVM,而且围绕玩安卓API(感谢鸿洋)带你们一块儿搭建咱们的MVVM项目。java

从零开始搭建MVVM架构系列文章(持续更新):
Android从零开始搭建MVVM架构(1)————DataBinding
Android从零开始搭建MVVM架构(2)————ViewModel
Android从零开始搭建MVVM架构(3)————LiveData
Android从零开始搭建MVVM架构(4)————Room(从入门到进阶)
Android从零开始搭建MVVM架构(5)————Lifecycles
Android从零开始搭建MVVM架构(6)————使用玩Android API带你搭建MVVM框架(初级篇)
Android从零开始搭建MVVM架构(7) ———— 使用玩Android API带你搭建MVVM框架(终极篇)react

在写文章以前我在想,为何MVP以后还有MVVM框架呢,咱们为何要用MVVM呢?android

在MVC衍变到MVP时。只是代码逻辑简洁了,View和Model也有解耦了,接手别人的项目时你的苦恼减小了。可是代价就是,接口爆炸,这也是作小项目的时候,根本不用它的缘由。那么今天咱们要讲的是MVVM。MVVM具有了MVP的优势外,并且不用像MVP那样写那么多接口了。ViewModel配合LiveData完成全部,又由于它具有生命周期,程序更健壮了,也不用写那么多判断代码了,更关键的是LiveData替代了那些接口,简直神奇了。git

这里我想说下个人感觉,框架的使用是我的的看法。每一个人都不一样。我这里搭建,是个人看法和想法。你能够借鉴个人思路,去搭建你本身的MVVM项目。github


1、建立一个新项目(本节将完成一个banner广告的功能)

打开DataBindingjson

//加载项目build.gradle的anroid标签下
dataBinding {
        enabled = true
    }
复制代码

添加相关依赖缓存

//okhttp、retrofit、rxjava
    implementation 'com.squareup.okhttp3:okhttp:3.8.0'
    implementation 'com.squareup.retrofit2:retrofit:2.3.0'
    implementation 'com.squareup.retrofit2:converter-gson:2.3.0'
    implementation 'com.squareup.retrofit2:adapter-rxjava2:2.3.0'
    implementation 'io.reactivex.rxjava2:rxjava:2.1.7'

    //放着没有及时回收形成RxJava内存泄漏
    implementation 'com.trello.rxlifecycle2:rxlifecycle-components:2.1.0'
    implementation 'android.arch.lifecycle:extensions:1.1.1'

    //Room的依赖引用
    implementation 'android.arch.persistence.room:runtime:2.1.4'
    annotationProcessor 'android.arch.persistence.room:compiler:2.1.4'

    //Room配合RxJava使用
    implementation 'android.arch.persistence.room:rxjava2:2.1.4'
    implementation 'io.reactivex.rxjava2:rxandroid:2.0.2'

    //广告banner
    implementation 'com.youth.banner:banner:1.4.10'

    //glide
    implementation 'com.github.bumptech.glide:glide:4.9.0'
    annotationProcessor 'com.github.bumptech.glide:compiler:4.9.0'
复制代码

2、建立咱们的Base

2.1 建立BaseViewModel

由于建立BaseActivity时,确定要引入咱们的BaseViewModel。因此咱们要先建立BaseViewModel。咱们知道,咱们要把公共代码和重复代码所有封装在咱们的Base里。固然这里咱们还不知道咱们的BaseViewModel要干吗,先建立吧,以后要什么,补什么网络

//继承AndroidViewModel,是由于里面要用context时候直接能够getApplication()
public abstract class BaseViewModel extends AndroidViewModel {

    public BaseViewModel(@NonNull Application application) {
        super(application);
    }

    @Override
    protected void onCleared() {
        super.onCleared();
        
    }
}
复制代码

2.2 建立BaseActivity

baseActivity里有2个引用,DataBinding 和 ViewModel,用泛型把他添加进来,架构

//ViewDataBinding 是全部DataBinding的父类
public abstract class BaseActivity<VM extends BaseViewModel, VDB extends ViewDataBinding> extends AppCompatActivity {   
    //获取当前activity布局文件,并初始化咱们的binding
    protected abstract int getContentViewId();
    
    //处理逻辑业务
    protected abstract void processLogic();


    protected VM mViewModel;
    protected VDB binding;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(getContentViewId());
        //初始化咱们的binging
        binding = DataBindingUtil.setContentView(this, getContentViewId());
        //给binding加上感知生命周期,AppCompatActivity就是lifeOwner,以前解释过了,不懂看前面
        binding.setLifecycleOwner(this);
        //建立咱们的ViewModel。
        createViewModel();
        processLogic();

    }

    public void createViewModel() {
        if (mViewModel == null) {
            Class modelClass;
            Type type = getClass().getGenericSuperclass();
            if (type instanceof ParameterizedType) {
                modelClass = (Class) ((ParameterizedType) type).getActualTypeArguments()[0];
            } else {
                //若是没有指定泛型参数,则默认使用BaseViewModel
                modelClass = BaseViewModel.class;
            }
            mViewModel = (VM) ViewModelProviders.of(this).get(modelClass);
        }
    }
}
复制代码

3、简单封装咱们的Retrofit

我这里只是简单封装咱们的Retrofit。本文终极篇demo RxJava + Retrofit联网(若是不熟悉,请看我另外一篇解读)RxJava + Retrofit + MVP(看完还不明白,吐槽我。适合初学者,VIP版MVP框架!!)
其中封装包括的内容有:app

  • 支持全部网络请求类型,get,post,put...(废话了!!Retrofit已经干了全部事情)
  • 支持上传文件并监听上传进度
  • 支持下载文件和断点下载并监听下载进度
  • 有网络时,支持在线缓存(链接网络时的有效期)
  • 断开网络,支持离线缓存(离线缓存有效期)
  • 屡次请求同一url,在网络还在请求时,是否只请求一次
  • 支持网络请求失败,自动重连
  • 支持取消网络请求

Retrofit的接口以下:

public interface RetrofitApiService {
    //wanAndroid的banner
    @GET("banner/json")
    Observable<ResponModel<List<BannerBean>>> getBanner();
}
复制代码

简单封装以下,封装一个单例的RetrofitManager:

public class RetrofitManager {
    private static RetrofitManager retrofitManager;
    private Retrofit retrofit;
    private RetrofitApiService retrofitApiService;
    private RetrofitManager() {
        initRetrofit();
    }
    
    public static RetrofitManager getInstance() {
        if (retrofitManager == null) {
            synchronized (RetrofitManager.class) {
                if (retrofitManager == null) {
                    retrofitManager = new RetrofitManager();
                }
            }
        }
        return retrofitManager;
    }
    
    public static RetrofitApiService getApiService() {
        return retrofitManager.retrofitApiService;
    }
    
    private void initRetrofit() {
        retrofit = new Retrofit.Builder()
                .baseUrl(SystemConst.DEFAULT_SERVER)
                .addConverterFactory(GsonConverterFactory.create())
                .addCallAdapterFactory(RxJava2CallAdapterFactory.create())
                .build();
        retrofitApiService = retrofit.create(RetrofitApiService.class);
    }
}
复制代码

4、实现咱们的功能

4.1 建立MainViewModel

首先是建立咱们的MainViewModel,并添加,加载banner接口。(Repository数据层,也能够说是model层,放在后一篇)以下:

public class MainViewModel extends BaseViewModel {

    public MainViewModel(@NonNull Application application) {
        super(application);
    }

    @Override
    protected void onCleared() {
        super.onCleared();

    }

    public MutableLiveData<List<BannerBean>> getBanners(){
        //由于用到LiveData,我以为都不须要切换到主线程了。LiveData能够帮咱们作
        //调用接口,返回咱们的MutableLiveData<List<BannerBean>>
        final MutableLiveData<List<BannerBean>> liveData = new MutableLiveData<>();
        RetrofitManager.getInstance().getApiService().getBanner()
                .subscribeOn(Schedulers.io())
                .subscribe(new Consumer<ResponModel<List<BannerBean>>>() {
                    @Override
                    public void accept(ResponModel<List<BannerBean>> listResponModel) throws Exception {
                        liveData.postValue(listResponModel.getData());
                    }
                }, new Consumer<Throwable>() {
                    @Override
                    public void accept(Throwable throwable) throws Exception {

                    }
                });
        
        return liveData;
    }
}
复制代码

4.2 MainActivity里

  • 首先把xml改为咱们的DataBinding布局
  • 添加一个按钮,点击去请求咱们的接口
  • 增长第三方Banner去展现咱们的数据

xml以下:

<layout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools">

    <data>

    </data>

    <RelativeLayout android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".MainActivity">
        <com.youth.banner.Banner android:id="@+id/banner" android:layout_width="match_parent" android:layout_height="180dp" />

        <Button android:text="点击请求" android:id="@+id/btn" android:layout_width="wrap_content" android:layout_height="wrap_content"/>
    </RelativeLayout>
</layout>
复制代码

MainActivity继承咱们的BaseActivity,并指明咱们的 ViewModel 和DataBinding。

public class MainActivity extends BaseActivity<MainViewModel, ActivityMainBinding> {

    @Override
    protected int getContentViewId() {
        return R.layout.activity_main;
    }

    @Override
    protected void processLogic() {
        initBanner();
        binding.btn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                getBanner();
            }
        });
    }

    private void getBanner() {
        mViewModel.getBanners().observe(this, new Observer<List<BannerBean>>() {
            @Override
            public void onChanged(List<BannerBean> bannerBeans) {
                updateBanner(bannerBeans);
            }
        });
    }


    private void initBanner() {
        binding.banner.setBannerStyle(BannerConfig.CIRCLE_INDICATOR_TITLE_INSIDE);
        //这是给banner添加图片加载器
        binding.banner.setImageLoader(new GlideImageLoader());
    }

    private void updateBanner(List<BannerBean> data) {
        if (data == null || data.size() <= 0) {
            return;
        }
        List<String> urls = new ArrayList<>();
        List<String> titles = new ArrayList<>();
        for (int i = 0; i < data.size(); i++) {
            urls.add(data.get(i).getImagePath());
            titles.add(data.get(i).getTitle());
        }
        binding.banner.setBannerTitles(titles);
        binding.banner.setImages(urls);
        binding.banner.start();
    }
}
复制代码

跟着项目走,你会跑通一个简单的MVVM项目。下一篇,将是最后终结篇。若是MVVM系列能帮到你的话,请帮楼主点个赞吧。谢谢

由于跟本文走,简单的MVVM会跑通。我这里就不贴demo连接了。下一篇,终结篇,将会放上终结篇demo连接。
最终运行的效果:



本文还涉及到的类有
ResponModel

public class ResponModel<T> implements Serializable {
    public static final int RESULT_SUCCESS = 0;

    private T data;
    private int errorCode;
    private String errorMsg;

    public T getData() {
        return data;
    }

    public void setData(T data) {
        this.data = data;
    }

    public int getErrorCode() {
        return errorCode;
    }

    public void setErrorCode(int errorCode) {
        this.errorCode = errorCode;
    }

    public String getErrorMsg() {
        return errorMsg;
    }

    public void setErrorMsg(String errorMsg) {
        this.errorMsg = errorMsg;
    }

    public boolean isSuccess(){
        return RESULT_SUCCESS == errorCode;
    }
}
复制代码

BannerBean:

public class BannerBean implements Serializable {

    /** * desc : Android高级进阶直播课免费学习 * id : 23 * imagePath : https://wanandroid.com/blogimgs/67c28e8c-2716-4b78-95d3-22cbde65d924.jpeg * isVisible : 1 * order : 0 * title : Android高级进阶直播课免费学习 * type : 0 * url : https://url.163.com/4bj */

    private String desc;
    private int id;
    private String imagePath;
    private int isVisible;
    private int order;
    private String title;
    private int type;
    private String url;

    public String getDesc() {
        return desc;
    }

    public void setDesc(String desc) {
        this.desc = desc;
    }

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getImagePath() {
        return imagePath;
    }

    public void setImagePath(String imagePath) {
        this.imagePath = imagePath;
    }

    public int getIsVisible() {
        return isVisible;
    }

    public void setIsVisible(int isVisible) {
        this.isVisible = isVisible;
    }

    public int getOrder() {
        return order;
    }

    public void setOrder(int order) {
        this.order = order;
    }

    public String getTitle() {
        return title;
    }

    public void setTitle(String title) {
        this.title = title;
    }

    public int getType() {
        return type;
    }

    public void setType(int type) {
        this.type = type;
    }

    public String getUrl() {
        return url;
    }

    public void setUrl(String url) {
        this.url = url;
    }
}
复制代码

GlideImageLoader:

public class GlideImageLoader extends ImageLoader {
    @Override
    public void displayImage(Context context, Object path, ImageView imageView) {
        Glide.with(context).load(path).placeholder(R.mipmap.ic_launcher)
                .error(R.mipmap.ic_launcher)
                .centerCrop().into(imageView);
    }
}
复制代码

最后别忘记加上网络权限。加油~(是否是发现接口被LiveData取代了呢!)

相关文章
相关标签/搜索