Android从零开始(第四篇)MVP + Retrofit + Rxjava

前言

终于到了网络请求模块了。因为以前一直使用的都是Okhttp,所以这次使用Retrofit的话花了两天时间去了解。总的来说Retrofit就是初始化搭建麻烦,挺多东西要注意的。不过看到各界大佬一致好评的请求框架,苦点就苦点。

走过路过点歌Start O(∩_∩)O
Github项目地址

这篇文章叫你如何搭建MVP+Retrofit+Rxjava,看完这篇文章你能学会:

  • Retrofit的基本使用
  • Retrofit + Rxjava + Mvp整合
    --------------------------------事不宜迟,上分割线------------------------------------------------
    这一片文章简单到不行,我都一度怀疑有没有必要写。

目录:
Retrofit使用思路

  1. 初始化环境
  2. 定义API
  3. 实例化API
  4. 处理响应定义API

Retrofit 如何结合Rxjava
Retrofit+Rxjava如何结合Mvp

Retrofit使用思路:

  • 跟Okhttp一样,要先先实例化嘛,做一些初始化环境
  • 定义API接口类(Retrofit是通过注解实现的)
  • 实例化API,调用接口就能发起请求
  • 对请求结果做处理

整体的思路其实跟OKhttp还是很相似的。
初始化环境:

/*
*   静态模式
* */

public class RetrofitUtil {

    /*
     *   BaseUrl
     * */
    private final static String BASE_URL = "http://v.juhe.cn/";

    private static Retrofit mRetrofit;

    private static RetrofitUtil mRetrofitUtil;

    static {
        mRetrofitUtil = new RetrofitUtil();
        initRetrofit();
    }

    private RetrofitUtil () {
    }

    public static RetrofitUtil newInstance() {
        return mRetrofitUtil;
    }

    /*
    *   初始化网络设置
    * */
    private static void initRetrofit () {
        // 设置okhttp
        OkHttpClient.Builder builder = new OkHttpClient.Builder()
                .connectTimeout(6000, TimeUnit.SECONDS)//设置超时时间
                .retryOnConnectionFailure(true);
        //添加拦截
        HttpLoggingInterceptor logInterceptor = new HttpLoggingInterceptor();
        builder.addInterceptor(logInterceptor);
        OkHttpClient client = builder.build();
        //设置retrofit
        mRetrofit = new Retrofit
                .Builder()
                .baseUrl(BASE_URL)
                .client(client)
                .addConverterFactory(GsonConverterFactory.create())
                .addCallAdapterFactory(RxJava2CallAdapterFactory.create())
                .build();
    }

    /*
    *   创建api
    * */
    public <T> T createApi (Class<T> cla) {
        return mRetrofit.create(cla);
    }


}

用了单例模式,这里暂时还没做很多通用的一些处理。完成这一步之后等于Retrofit的环境已经创建好了。这个时候就要开始定义具体的请求,通过这个retrofit对象来发起。接着看:
这里要注意的是:BASE_URL可是有要求的,不能随便写;例如后面的路径不能追加在这里。

忘记贴依赖包了:按需添加

// 网络请求 Retrofit + Rxjava
    implementation 'com.squareup.retrofit2:retrofit:2.3.0'//导入retrofit
    implementation 'com.google.code.gson:gson:2.6.2'//Gson 库
    implementation 'com.squareup.okhttp3:okhttp:3.1.2'
    implementation 'com.squareup.okhttp3:logging-interceptor:3.1.2'
    implementation 'com.orhanobut:logger:2.1.0' // 打印日志
    //下面两个是RxJava 和 RxAndroid
    implementation 'io.reactivex.rxjava2:rxandroid:2.0.2'
    implementation 'io.reactivex.rxjava2:rxjava:2.x.y'
    implementation 'com.squareup.retrofit2:converter-gson:2.3.0'//转换器,请求结果转换成Model
    implementation 'com.squareup.retrofit2:adapter-rxjava2:2.3.0'//配合Rxjava

定义API接口:

public interface Api {

    //http://v.juhe.cn/toutiao/index?type=top&key=APPKEY
    @GET("/toutiao/index")
    Observable<NewsDomain> getNews (@Query("type")String type, @Query("key")String apiKey);

}

这些注解的用法使得代码很简洁,只需要多看看官网/教程就知道怎么用了。

实例化API

public class TestModel implements ITest {
    private static final String TAG = "bigname";

    @Override
    public Observable<NewsDomain> test() {
        RetrofitUtil retrofitUtil = RetrofitUtil.newInstance();
        Api api = retrofitUtil.createApi(Api.class);
        return api.getNews("top", "dc4fed8fb5fb46ac78e51d66306e6762");
    }
}

到这里,应用就能够发起请求了。

处理请求响应

public class MainPresenter extends MainContract.MainPresenter {
    private static final String TAG = "bigname";

    @Override
    public void login(String name, String password) {
        TestModel testModel = new TestModel();
        Observable<NewsDomain> newsDomainObservable = testModel.test();
        newsDomainObservable.subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new DisposableObserver<NewsDomain>() {
                    @Override
                    public void onNext(NewsDomain newsDomain) {
                        Log.d(TAG, "onNext: " +  newsDomain.toString());
                        mView.showTips(newsDomain.getReason());
                    }

                    @Override
                    public void onError(Throwable e) {
                        Log.d(TAG, "onError: " + e);
                    }

                    @Override
                    public void onComplete() {
                        Log.d(TAG, "onComplete: ");
                    }
                });
    }
}

就是这么简单。
看看效果:


6423516-dd8bbfa97c18129a.gif
在这里插入图片描述

如果出现报错:java.net.UnknownServiceException
这是因为API28开始网络请求更加严格引起的,解决方法:
1.在res目录下新建xml资源,然后创建network_security_config.xml文件,
内容:

<?xml version="1.0" encoding="utf-8"?>
<network-security-config>
    <base-config cleartextTrafficPermitted="true" />
</network-security-config>

2.在清单文件中:

<application
        android:networkSecurityConfig="@xml/network_security_config"

这样就解决了。

------------------------------------人工分割-------------------------------------------

Retrofit + Rxjava怎么实现?

这个我刚开始以为还多麻烦呢。
Retrofit嵌入Rxjava只需简单几步:

  • 导入依赖包rxjava(上面贴的已经在内)
  • retrofit初始化增加对rxjava的支持
  • 请求响应从Call变成Obserable

依赖就不贴图了,
retrofit对rxjava支持:

mRetrofit = new Retrofit
                .addCallAdapterFactory(RxJava2CallAdapterFactory.create())
                .build();

要注意的是Rxjava2CallAdapterFactroy()在依赖版本方面需要注意,要对应的。
请求响应从Call变成Obserable

@GET("/toutiao/index")
    Observable<NewsDomain> getNews (@Query("type")String type, @Query("key")String apiKey);

如果不适用Rxjava,则返回Call对象,这种方式一样能实现功能。

-------------------------------------------人工分割线---------------------------------------------------

Mvp如何结合进来:

简单到真的没必要打字的感觉。传统的方式是构造ICallBack传递给Model层,然后利用接口回调的方式,现在这种是Model层会返回Obserable观察者对象,Presenter层拿到这个Obserable对象之后把他当成CallBack对象来使用就行了。

Observable<NewsDomain> newsDomainObservable = testModel.test();
        newsDomainObservable.subscribeOn(Schedulers.io())...

The end...

猜你喜欢

转载自blog.csdn.net/weixin_34417200/article/details/87499490