优雅的构建Android项目之RxAndroid+Retrofit网络请求

注意

Retrofit 2.0+和Retrofit 2.0之前的版本语法上有差别,本文基于Retrofit2.1.0

什么是Retrofit?

retrofit是一款针对Android网络请求的开源框架,它与okhttp一样出自Square公司。Rotrofit2.0的网络框架全部交给了okhttp来实现,Android N之后Apache的httpclient已经被Google从SDK中移除,Okhttp则成功上位。Retrofit的网络请求实现风格与URLconnection和httpClient有较大的差别。创建请求的时候需要先创建基于注解的服务接口(不了解的可以先了解一下注解),进行网络请求的时候再通过retrofit.creat()方法创建请求。

Retrofit中http POST/GET请求

Retrofit中的网络请求都是通过注解方式的接口方法来表示的,此处只对常用的post和get请求进行说明,Retrofit还提供有put,delete等请求方式可自己研究文档使用

post请求

  • Body对象作为post参数
@POST("user/login")
Call<User> login(@Body LoginInfo loginInfo);
  • Field方式
@FormUrlEncoded
@POST("user/login")
Call<User> login(@Field("username") String username,
                 @Field(password) String password);
  • FieldMap方式
@FormUrlEncoded
@POST("user/login")
Call<User> login(@FieldMap Map<String,String> map);

参数较多时建议用Body方式和FieldMap方式

get请求

  • 直接请求某一地址获取列表
//接口我瞎写的
@GET("news/toplist")
Call<ArrayList<News> news> getNewsList(); 
  • url拼接固定查询条件
@GET("news/toplist?date=20161030")
Call<ArrayList<News> news> getNewsList(); 
  • url中拼接地址信息
@GEt("news/{city}/newslist")
Call<ArrayList<News> news> getCityNewsList(@Path("city") String city);
  • 通过Query注解添加其他查询条件
@GET("news/{city}/newslist")
Call<ArrayList<News> news> getCityNewsList(@Path("city") String city
                                           @Query("date") String date
                                           @Query("newsType") String newsType);
  • 查询条件较多时同样有QueryMap注解方法供使用
@GET("news/{city}/newslist")
Call<ArrayList<News> news> getCityNewsList(@Path("city") String city
                                           @QueryMap<String, String> options);

通过上面的API方法会发现都是在进行请求条件的配置,假如我要给请求加请求头怎么办?放心,retrofit也有相应的注解。除了注解之外还有一个万用的处理方法。

Header请求头设置

  • 为请求添加固定请求头
//添加单个固定请求头
@Header("Cache-Control: max-aget-640000")
@GET("news/toplist?date=20161030")
Call<ArrayList<News> news> getNewsList(); 
//多个请求头以数组的形式提交
@Header(
    {"Accept: application/vnd.github.v3.full+json",
    "User-Agent: Retrofit-Sample-App"
    })
@GET("news/toplist?date=20161030")
Call<ArrayList<News> news> getNewsList(); 
  • 动态添加请求头
//添加动态请求头,比如获取的认证信息等
@GET("news/toplist?date=20161030")
Call<ArrayList<News> news> getNewsList(@Header(Authorization) String authorization); 

上面的两种添加请求头的方法作用范围只是添加注解的单个方法,如果想为每个请求都添加请求头还按这种方式来做的话就很不程序猿了。Retrofit的网络请求全部交给okhttp来处理,因此我们可以通过OkHttpClient来做文章,自己重写okhttp的拦截器在拦截器内再进行需要的操作

Okhttp interceptor

拦截器顾名思义,所有通过okhttp进行的请求都会过一遍okhttpClient的拦截器,发出去的请求,收到的响应都会经过他,就像一个双向的安检通道。
okhttp拦截器的原理如下:

okhttp拦截器,图片来自okhttp官网
okhttp拦截器,图片来自okhttp官网

如图所示拦截器分为Application Interceptors和NetWork Interceptors。Application拦截器工作区域为应用发出请求到okhttp核心之间,远端响应经过okhttp核心后到达应用处理之前。而NetWork拦截器的作用域为okhtt核心到远端服务器之间的部分。明显区别就是当一次请求中会有一个重定向的时候Application拦截器只会响应一次,因为对于应用来说就进行了一次请求。而NetWork拦截器会在重定向时也响应即响应两次,也不难理解,毕竟重定向也会经过一次okhttp核心嘛。

拦截器工作示意简图
拦截器工作示意简图

上图是okhttp拦截器工作原理简图,重点在右边部分。当多个拦截器配合使用时,不用担心请求拦截和响应拦截顺序会错乱,okhttp已经给你排好了。


上传个需要压缩和编码的东东的时候,你可以选择先写个拦截器请求时压缩响应时解压,再写个拦截器请求时编码响应时解码。加起来就是压缩->编码->okhttClient与服务器的Py交易->解码->解压跟栈先进后出类似。

原理扯了一大堆,代码才是干货,看了代码才知道怎么用。

//官方的栗子
class LoggingInterceptor implements Interceptor {
        @Override
        public Response intercept(Interceptor.Chain chain) throws IOException {
         //拿到request实例在此对请求做需要的设置
            Request request = chain.request();
            long t1 = System.nanoTime();
            logger.info(String.format("Sending request %s on %s%n%s",
                    request.url(), chain.connection(), request.headers()));
            //发送request请求
            Response response = chain.proceed(request);
            //得到请求后的response实例,做相应操作
            long t2 = System.nanoTime();
            logger.info(String.format("Received response for %s in %.1fms%n%s",
                    response.request().url(), (t2 - t1) / 1e6d, response.headers()));
            return response;
        }
    }

通过Request request = chain.request();拿到请求实例,想怎么装扮就怎么装扮,什么加请求头,设置编码格式soeasy。前面说到的为每个请求设置请求头就是在这完成设置工作的。但是真正要加到请求里跟retrofit的ApiService接口一起用还需要将Okhttp注册拦截器后与Retrofit绑定才行。

//注册应用拦截器
OkHttpClient client = new OkHttpClient.Builder()
    .addInterceptor(new LoggingInterceptor())
    .build();
Request request = new Request.Builder()
    .url("http://www.publicobject.com/helloworld.txt")
    .header("User-Agent", "OkHttp Example")
    .build();··
Response response = client.newCall(request).execute();
response.body().close();

//注册网络拦截器
OkHttpClient client = new OkHttpClient.Builder()
    .addNetworkInterceptor(new LoggingInterceptor())
    .build();
Request request = new Request.Builder()
    .url("http://www.publicobject.com/helloworld.txt")
    .header("User-Agent", "OkHttp Example")
    .build();
Response response = client.newCall(request).execute();
response.body().close();

该如何使用Retrofit?

基本也说得差不多了,那么怎么使用Retrofit进行一次完整的网络请求呢
需要注意一下Retrofit的Url拼接规则

enter description here
enter description here

enter description here
enter description here

enter description here
enter description here

个人建议以第一幅图的方式,baseUrl总是以/结尾,接口rul总是不以/开头

  • 1、当然是引入retrofit的库啦
//build.gradle的依赖中加入,其中第二条不一定要使用gson,其他方式在官方的github上也有
// retrofit
compile 'com.squareup.retrofit2:retrofit:2.1.0'
compile 'com.squareup.retrofit2:converter-gson:2.1.0'
compile 'com.squareup.retrofit2:adapter-rxjava:2.1.0'
  • 2、创建一个ServiceApi接口
//方便后面RxAndroid我把RxAndroid方式的接口也贴上来。只有返回类型不同而已
public interface RetrofitService {
    //单纯使用retrofit接口定义
    @GET("news/latest")
    Call<ZhiHuDaily> getZhihuDailyRetrofitOnly();

    //使用retrofit+RxAndroid的接口定义
    @GET("news/latest")
    Observable<ZhiHuDaily> getZhihuDaily();
}
  • 3、我建议是维护一个统一的api管理类。当然你要直接拿接口用也行,但可维护性会降低很多
public class ApiManager {

    private RetrofitService mDailyApi;
    private static ApiManager sApiManager;
    //获取ApiManager的单例
    public static ApiManager getInstence() {
        if (sApiManager == null) {
            synchronized (ApiManager.class) {
                if (sApiManager == null) {
                    sApiManager = new ApiManager();
                }
            }
        }
        return sApiManager;
    }
    /**
     * 封装配置知乎API
     */
    public RetrofitService getDailyService() {
    //不需要使用拦截器就不创建直接从if开始
        OkHttpClient client = new OkHttpClient.Builder()
                //添加应用拦截器
                .addInterceptor(new MyOkhttpInterceptor())
                //添加网络拦截器
//                .addNetworkInterceptor(new MyOkhttpInterceptor())
                .build();
        if (mDailyApi == null) {
            Retrofit retrofit = new Retrofit.Builder()
                    .baseUrl(GlobalConfig.baseUrl)
                    //将client与retrofit关联
                    .client(client)
                    .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
                    .addConverterFactory(GsonConverterFactory.create())
                    .build();
            //到这一步创建完成
            mDailyApi = retrofit.create(RetrofitService.class);
        }
        return mDailyApi;
    }
}
  • 4、调用接口方法进行网络请求
    public void getStoryDataByRetrofit(final OnEventLister<ArrayList<ZhihuStory>> eventLister) {
        ApiManager apiManager = ApiManager.getInstence();
        Call<ZhiHuDaily> call = apiManager.getDailyService().getZhihuDailyRetrofitOnly();
        //发送异步请求
        call.enqueue(new Callback<ZhiHuDaily>() {
            @Override
            public void onResponse(Call<ZhiHuDaily> call, Response<ZhiHuDaily> response) {
                eventLister.onSuccess(response.body().getStories());
            }

            @Override
            public void onFailure(Call<ZhiHuDaily> call, Throwable t) {
                eventLister.onFail(t.getMessage(), "");
            }
        });
    }

使用Retrofit的好处

  • 可以少写不少的代码
  • 接口方便维护需要改什么直接到ApiService中进行配置即可
  • 异步请求不再需要自己来newThread再handler,也不需要自己再来写请求结果回调。异步请求只需要使用call.enqueue()即可。
  • 支持RxAndroid,这个我觉得很重要
  • 降低工程的耦合度,网络请求跟逻辑代码完全剥离开。需要的仅仅是传递参数有的请求甚至参数都不需要传递。直接在接口中配置就好。

Retrofit的好基友——RxAndroid

RxAndroid是RxJava在Android上的变种。那么RxJava到底是什么呢?
"a library for composing asynchronous and event-based programs using observable sequences for the Java VM"(一个在 Java VM 上使用可观测的序列来组成异步的、基于事件的程序的库)。这是github项目主页的自我概括,我觉得其实就两个关键词,异步基于事件。这里我只说一下RxAndroid怎么跟Retrofit搭配使用,要进一步了解可以异步扔物线大神的文章给 Android 开发者的 RxJava 详解。讲的肯定比我好,我就是看这个入门的。

Retrofit+RxAndroid使用

因为用到Retrofit所以定义接口,创建ApiManager这些跟上面单纯用Retrofit是一毛一样的,唯一的不同是接口的返回类型从Retrofit的Call对象变成了Observable对象,即被观察者对象。然后就是调用进行网络请求部分变成如下形式

    //使用rxandroid+retrofit进行请求
    public void loadDataByRxandroidRetrofit() {
        mIMainActivity.showProgressBar();
        Subscription subscription = ApiManager.getInstence().getDailyService()
                .getZhihuDaily()
                .map(new Func1<ZhiHuDaily, ArrayList<ZhihuStory>>() {
                    @Override
                    public ArrayList<ZhihuStory> call(ZhiHuDaily zhiHuDaily) {
                        return zhiHuDaily.getStories();
                    }
                })
                //设置事件触发在非主线程
                .subscribeOn(Schedulers.io())
                //设置事件接受在UI线程以达到UI显示的目的
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Subscriber<ArrayList<ZhihuStory>>() {
                    @Override
                    public void onCompleted() {
                        mIMainActivity.hidProgressBar();
                    }

                    @Override
                    public void onError(Throwable e) {
                        mIMainActivity.getDataFail("", e.getMessage());
                    }

                    @Override
                    public void onNext(ArrayList<ZhihuStory> stories) {
                        mIMainActivity.getDataSuccess(stories);
                    }
                });
        //绑定观察对象,注意在界面的ondestory或者onpouse方法中调用presenter.unsubcription();进行解绑避免内存泄漏
        addSubscription(subscription);
    }

网络请求得到的是一个Obserable对象,该对象再通过subscrible()绑定一个观察者对象,观察者对象中有onCompleted(),onError(),onNext()三个回调方法。事件过程中出错onError()触发并停止后续事件,一个Obserable对象一次发出多个事件每次都会触发onNext(),当不再有事件发出的时候onCompleted()方法触发并结束。异步在RxAndroid中变得很简单subscribeOn指定事件发生线程,如上面的网络请求被指定在io线程中,observeOn指定事件的消费线程,如上面的知乎故事数据结果被交给主线程显示。

RxAndroid的优点

看完上面单独使用retrofit和使用retrofit+rxandroid两种方式之后你也许会吐槽,尼玛脑子有坑?明明代码变多了。
但是你也会发现代码中都是.XX()的形式如果需求变化更多一些会更明显。比如说就上面的例子我要在每一条信息中修改某一个值。并且又要对结果进行一些筛选。只用retrofit的话是不是应该向这样:

    call.enqueue(new Callback<ZhiHuDaily>() {
        @Override
        public void onResponse(Call<ZhiHuDaily> call, Response<ZhiHuDaily> response) {
            ArrayList<ZhihuStory> stories = response.body().getStories();
            for(ZhihuStory story : stories){
                //修改每一条story中的某一值,这里用XXX代替
                story.setXXX(XXX);
                //筛选出id<100的
                if(story.getId()>100){
                    stories.remove(story);
                }
            }
            eventLister.onSuccess(response.body().getStories());
        }

        @Override
        public void onFailure(Call<ZhiHuDaily> call, Throwable t) {
            eventLister.onFail(t.getMessage(), "");
        }
    });

如果需要设置和条件赛选层次越多会发现for套if,if再if会越嵌套越多。隔一段时间之后就真的成了“当初写下这段代码的时候只有我跟上帝知道他是干嘛的,现在只有上帝知道他是干嘛的”。而使用RxAndroid的话整个变换过程都是线性的哪一步做了什么都会很清楚不会出现各种蜜汁缩进:

        Subscription subscription = ApiManager.getInstence().getDailyService()
                .getZhihuDaily()
                //从ZhihuDaily中获取Stories列表
                .map(new Func1<ZhiHuDaily, ArrayList<ZhihuStory>>() {
                    @Override
                    public ArrayList<ZhihuStory> call(ZhiHuDaily zhiHuDaily) {
                        return zhiHuDaily.getStories();
                    }
                })
                //将列表拆开成事件发送
                .flatMap(new Func1<ArrayList<ZhihuStory>, Observable<ZhihuStory>>() {
                    @Override
                    public Observable<ZhihuStory> call(ArrayList<ZhihuStory> stories) {
                        return Observable.from(stories);
                    }
                })
                //将story中的XXX设置为xxx
                .map(new Func1<ZhihuStory, ZhihuStory>() {
                    @Override
                    public ZhihuStory call(ZhihuStory zhihuStory) {
                        zhihuStory.setXXX(xxx);
                        return zhihuStory;
                    }
                })
                //过滤掉Id>10的story
                .filter(new Func1<ZhihuStory, Boolean>() {
                    @Override
                    public Boolean call(ZhihuStory zhihuStory) {
                        return zhihuStory.getId()<10;
                    }
                })
                //将结果重新整理成List
                .toList()
                //设置事件触发在非主线程
                .subscribeOn(Schedulers.io())
                //设置事件接受在UI线程以达到UI显示的目的
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Subscriber<List<ZhihuStory>>() {
                    @Override
                    public void onCompleted() {
                        mIMainActivity.hidProgressBar();
                    }

                    @Override
                    public void onError(Throwable e) {
                        mIMainActivity.getDataFail("", e.getMessage());
                    }

                    @Override
                    public void onNext(List<ZhihuStory> stories) {
                        mIMainActivity.getDataSuccess((ArrayList<ZhihuStory>) stories);
                    }
                });

越复杂的逻辑,Rx的优势也就越明显。Rx的操作符各种组合起来几乎能够满足所有的变换需求。开始写可能会觉得很不适应,但熟练使用之后会默念Rx大法好的。
对于RxJava操作符鼠标悬停都会有文字和示意图的,另外发现一个不错的博客里面也有较详细的解析RxJava/RxAndroid操作符

Demo地址

在之前的那个MVPDemo的基础上写的,okhttp请求方式,单纯retrofit方式,retrofit+rxAndroid方式的请求都有保留,可以对比着感受一下。
demo仓库地址

觉得本文对你有帮助

关注简书PandaQ404,持续分享中。。。
Github主页

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 160,227评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,755评论 1 298
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,899评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,257评论 0 213
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,617评论 3 288
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,757评论 1 221
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,982评论 2 315
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,715评论 0 204
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,454评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,666评论 2 249
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,148评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,512评论 3 258
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,156评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,112评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,896评论 0 198
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,809评论 2 279
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,691评论 2 272

推荐阅读更多精彩内容