//加载项目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’
二、创建我们的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); } } }
三、简单封装我们的Retrofit
我这里只是简单封装我们的Retrofit。本文终极篇demo RxJava + Retrofit联网(如果不熟悉,请看我另一篇解读)RxJava + Retrofit + MVP(看完还不明白,吐槽我。适合初学者,VIP版MVP框架!!) 其中封装包括的内容有:
- 支持所有网络请求类型,get,post,put…(废话了!!Retrofit已经干了所有事情)
- 支持上传文件并监听上传进度
- 支持下载文件和断点下载并监听下载进度
- 有网络时,支持在线缓存(连接网络时的有效期)
- 断开网络,支持离线缓存(离线缓存有效期)
- 多次请求同一url,在网络还在请求时,是否只请求一次
- 支持网络请求失败,自动重连
- 支持取消网络请求
Retrofit的接口如下:
public interface RetrofitApiService { //wanAndroid的banner @GET(“banner/json”) Observable<ResponModel<List>> 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.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> getBanners(){ //因为用到LiveData,我觉得都不需要切换到主线程了。LiveData可以帮我们做 //调用接口,返回我们的MutableLiveData<List> final MutableLiveData<List> liveData = new MutableLiveData<>(); RetrofitManager.getInstance().getApiService().getBanner() .subscribeOn(Schedulers.io()) .subscribe(new Consumer<ResponModel<List>>() { @Override public void accept(ResponModel<List> listResponModel) throws Exception { liveData.postValue(listResponModel.getData()); } }, new Consumer() { @Override public void accept(Throwable throwable) throws Exception {
} });
return liveData; } }
4.2 MainActivity里
- 首先把xml改成我们的DataBinding布局
- 添加一个按钮,点击去请求我们的接口
- 增加第三方Banner去展示我们的数据
xml如下:
<com.youth.banner.Banner android:id="@+id/banner" android:layout_width=“match_parent” android:layout_height=“180dp” />
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>() { @Override public void onChanged(List bannerBeans) { updateBanner(bannerBeans); } }); }
private void initBanner() { binding.banner.setBannerStyle(BannerConfig.CIRCLE_INDICATOR_TITLE_INSIDE); //这是给banner添加图片加载器 binding.banner.setImageLoader(new GlideImageLoader()); }
private void updateBanner(List data) {
|