Retrofit项目封装使用

欢迎大家访问我的博客:博客地址

概述

  1. Retrofit开源项目地址
  2. Retrofit项目官网

官网上有它的一系列基本用法,以及他的介绍:A type-safe HTTP client for Android and Java,它和Okhttp一样都是square公司的开源项目。

RESTful Api:

看下面设计的三个删除评论的 api

http://test.net/?method=comment.del&id=x

http://test.net/comment/del/id/x

而 RESTful Api 则是:

[DELETE] http://test.net/comments/1

我们对比可以发现①和② URL 中,都有del的动作指示。

SOAP Web API采用RPC风格,它采用面向功能的架构,所以我们在设计SOAP Web API的时候首相考虑的是应高提供怎样的功能(或者操作)。而 RESTful Api 是面向资源的架构。是查询、新增、修改、删除,都与该资源无关。

RESTful Api 是以 HTTP 协议为强烈依托的,将类似于①和②这种以功能为主导的URL风格舍弃,还原 URL 的本质,它的宗旨就是一个 URL 就应该是一个资源,不能包含任何动作,如下所示:

- [POST]     http://test.net/users   // 新增
- [GET]      http://test.net/users/1 // 查询
- [PATCH]    http://test.net/users/1 // 更新
- [PUT]      http://test.net/users/1 // 覆盖,全部更新
- [DELETE]   http://test.net/users/1 // 删除

url的简单构成

构成一般是这样的:[scheme:][//authority][path][?query]

看下面一个url:

http://www.java2s.com:8080/yourpath/fileName.htm?stove=10&path=32&id=4
  • scheme: http

  • authority: www.java2s.com:8080

  • path: /yourpath/fileName.htm

  • query:在?后的部分为:stove=10&path=32&id=4

  • 又由于authority又一步可以划分为host:port形式,其中host:port用冒号分隔,冒号前的是host,冒号后的是port,所以:

    host:www.java2s.com

    port:8080

这里是url中的参数,除了url中的参数还有两个http协议中常用参数是:header(请求头)和body(常用于post请求中的请求体,有多种封装方法,不暴露在url中)这两个参数。

可以看出整个网络请求中参数主要可以分成:scheme、authority、path、query、header、body这六块,下面主要看下Retrofit怎么配置这六块参数的。

Retrofit参数配置

scheme和authority

在retrofit中将这两者合体称为baseurl,接口用法如下:

Retrofit retrofit = new Retrofit.Builder()
    .baseUrl("https://api.github.com/")
    .build();

header

  • 静态header(分为单个键值对和多个键值对两种注解方式):

单个键值对注解:

@Headers("Cache-Control: max-age=640000")
@GET("widget/list")
Call<List<Widget>> widgetList();

多个键值对注解:

@Headers({
    "Accept: application/vnd.github.v3.full+json",
    "User-Agent: Retrofit-Sample-App"
})
@GET("users/{username}")
Call<User> getUser(@Path("username") String username);
  • 动态header(分为局部动态header和全局动态header)

局部动态header(分为单个键值对header和多个键值对header):

单个键值对局部动态header:

@GET("user")
Call<User> getUser(@Header("Authorization") String authorization)

多个键值对局部动态header(retrofit:2.1.0新加的):

@GET("user")
Call<User> getUser(@HeaderMap Map<String, String> headerMap)

全局动态header(适用于项目中的header规则一致的情况)

这里是用了 OkHttp的interceptor:

private static void initHttpClient() {
        OkHttpClient.Builder httpClientBuilder = new OkHttpClient.Builder();
        if (httpClientBuilder.interceptors() != null) {
            httpClientBuilder.interceptors().clear();
        }
        httpClientBuilder.addInterceptor(new Interceptor() {
            @Override
            public Response intercept(Chain chain) throws IOException {
            
                //这里可以获取到请求的request所有数据
                Request request = chain.request();
                String path = request.url().encodedPath();
                Log.d("AppClient", path + ">>>path");
                String query = request.url().query();
                if (BuildConfig.DEBUG){
                    Log.d("AppClient", query + ">>>query");
                }
                //这里设置成你的全局header
                Request interRequest = chain.request().newBuilder()
                        .headers(Headers.of(Map yourHeader))
                        .build();
                return chain.proceed(interRequest);
            }
        })
                .connectTimeout(10, TimeUnit.SECONDS)
                .readTimeout(10, TimeUnit.SECONDS);
        
        mOkHttpClient = httpClientBuilder.build();
    }

path

这里只看下GET请求,POST, PUT, DELETE请求一样处理

这里分为带参数和不带参数两种:

  1. 不带参数的:
@GET("widget/list")
Call<List<Widget>> widgetList();
  1. 带参数的:
@GET("users/{user}/repos")
Call<List<Repo>> listRepos(@Path("user") String user);

这里是通过@Path("user")注解实现参数传递。

query

以GET请求为例提供了两种注解方式(单个键值对和多个键值对),如下:

单个键值对:

@GET("group/{id}/users")
Call<List<User>> groupList(@Path("id") int groupId, @Query("sort") String sort);

多个键值对:

@GET("group/{id}/users")
Call<List<User>> groupList(@Path("id") int groupId, @QueryMap Map<String, String> options);

body

用于POST请求中:

@POST("users/new")
Call<User> createUser(@Body User user);

Retrofit请求简单封装

写在前面:Retrofit中网络请求实际上用的还是他自家的OkHttp,所以如果要配置请求的一些全局参数还是得先创建一个OkHttp的实例对象,然后一一配置,上面讲了配置全局header也是这样处理的,所以这里我将其使用分成了三步:

这里首先要配三个库(retrofit库,json转化库,log的截断器库):

compile 'com.squareup.retrofit2:retrofit:2.1.0'
compile 'com.squareup.retrofit2:converter-gson:2.1.0'
compile 'com.squareup.okhttp3:logging-interceptor:3.4.0-RC1'
  1. 构建OkHttpClient实例(配置一些请求的全局参数):
 private static void initHttpClient() {
        OkHttpClient.Builder httpClientBuilder = new OkHttpClient.Builder();
        httpClientBuilder.hostnameVerifier(new HostnameVerifier() {
            @Override
            public boolean verify(String hostname, SSLSession session) {
                return true;
            }
        });
        //处理拦截器,主要是做了个header和连接超时、读取超时设置,我项目里header放了些签名信息,主要是这里能拿到整个请求的所有参数,做任何想做的事,而且是全局动态处理
        if (httpClientBuilder.interceptors() != null) {
            httpClientBuilder.interceptors().clear();
        }
        httpClientBuilder.addInterceptor(new Interceptor() {
            @Override
            public Response intercept(Chain chain) throws IOException {
                Request request = chain.request();
                String path = request.url().encodedPath();
                Log.d("AppClient", path + ">>>path");
                String query = request.url().query();
                if (BuildConfig.DEBUG){
                    Log.d("AppClient", query + ">>>query");
                }
                Map<String, Object> queryParam = null;
                if (query != null){
                    queryParam = new HashMap();
                    String queryEntries[] = query.split("&");
                    for (int i = 0; i < queryEntries.length; i ++){
                        queryParam.put(queryEntries[i].split("=")[0], queryEntries[i].split("=")[1]);
                    }
                }
                if (BuildConfig.DEBUG){
                    Log.d("AppClient", "queryParam:" + queryParam + ">>>queryParam");
                }
                String signature = makeSignature_v3(path, queryParam);

                Request interRequest = chain.request().newBuilder()
                        .headers(Headers.of(getHeaders(signature)))
                        .build();
                return chain.proceed(interRequest);
            }
        })
                .connectTimeout(10, TimeUnit.SECONDS)
                .readTimeout(10, TimeUnit.SECONDS);
         //在debug模式下我使用了一个他家公司的一个log拦截器
        if (BuildConfig.DEBUG) {
            HttpLoggingInterceptor loggingInterceptor = new HttpLoggingInterceptor();
            loggingInterceptor.setLevel(HttpLoggingInterceptor.Level.BODY);
            httpClientBuilder.addInterceptor(loggingInterceptor);
        }
        //通过build模式构建实例
        mOkHttpClient = httpClientBuilder.build();
    }

这里做了四件事:

  • httpClientBuilder.hostnameVerifier这里是针对我公司的Https证书问题,详情看这里:https证书问题

  • 通过拦截器设置header,可以看上面参数设置那里的讲解

  • 设置请求超时时间和读取数据时间

.connectTimeout(10, TimeUnit.SECONDS)
.readTimeout(10, TimeUnit.SECONDS);
  • 设置log拦截器,这里设置之后可以在logcat中看到每个网络请求的请求参数和返回结果以及请求时长,非常好用,当然先要配置下上面我配置的第三个库;
  1. 构建retrofit实例:
public static Retrofit retrofit() {
        if (mRetrofit == null) {
            initHttpClient();
            mRetrofit = new Retrofit.Builder()
                    .baseUrl(BASE_URL)
                    .client(mOkHttpClient)
                    .addConverterFactory(GsonConverterFactory.create())                 .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
                    .build();
        }
        return mRetrofit;
    }

这里配置可以看出,Retrofit内部也是封了一个Okhttp的实例,这里做了四件事:

  • 设置了baseurl参数,可以看我上面参数配置的讲解;

  • 设置了OkHttpClient实例;

  • gson转化工厂:addConverterFactory(GsonConverterFactory.create()),这里Retrofit内部会根据这个转换工厂及返回数据所指定的泛型实现直接转换;

  • 网络请求的适配器工厂:addCallAdapterFactory(RxJavaCallAdapterFactory.create()),这个可以忽略暂时,因为我项目里使用了RXjava和Retrofit,后续我会写一篇MVP+RXjava+Retrofit的封装使用

  1. 构建call,开始网络请求并实现回调:

这里先要配置请求参数,我封了个类专门用来设置请求接口:

public interface ApiStores {
    //POST登录
    String RES_USERS_LOGIN = "users/login";
    //POST注册
    String RES_USERS_REGISTER = "users";

    @POST(RES_USERS_LOGIN)
    Observable<Map> login(@Body Map<String, String> body);
    
    @POST(RES_USERS_REGISTER)
    Observable<Map> register(@Body Map<String, String> body);

}

构建call实例对象,开始网络请求并实现回调:

public void login(String cityId, final String cityName, final String mobile, String password, final OnLoginListener listener) {
        Map<String, String> body = new HashMap<>();
        body.put("mobile", mobile);
        body.put("password", password);
        body.put("city", cityId);
        body.put("embed", "home");       
        ApiStores apiStores = AppClient.retrofit().create(ApiStores.class);
        //这里通过指定Map泛型集合上面的Gson转换工厂实现返回数据json转换成Map
        Call<Map> call = apiStores.login(body);
        call.enqueue(new Callback<Map>() {
            @Override
            public void onResponse(Call<Map> call, Response<Map> response) {
                //请求成功做的事情,这里两个参数:call是请求时候的call实例,可以拿到请求的request实例,response是服务端返回的参数,里面包含了code和body(这里的body类型是通过call指定的泛型和Gson转换工厂实现的),message一系列数据。
            }

            @Override
            public void onFailure(Call<Map> call, Throwable t) {
                //请求失败做的事情,call和上面一样,外加一个Throwable
            }
        });

小结

Retrofit这个库将ResfulApi的几种请求都通过注解进行了一系列封装,整个请求流程比OkHttp简洁了很多,同时它可以通过设置GSON转换工厂内部实现返回数据的转化,使用过程非常简洁清晰,同时可以配合现在流行的RXjava一起使用,下面我准备搞一篇MVP+RX+Retrofit的封装使用,Okhttp可以换啦!

具体代码请看我的github:Android练习小项目

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,478评论 25 707
  • 准备工作 1. 了解RESTful Api: 看下面设计的三个删除评论的 api http://test.net/...
    jacky123阅读 272评论 0 0
  • 整体Retrofit内容如下: 1、Retrofit解析1之前哨站——理解RESTful2、Retrofit解析2...
    隔壁老李头阅读 15,026评论 4 39
  • #微写作#004 原来反思要从每一件小事做起,我们的任何一个举动都是我们想法的直接感受从而造成的直接做法。那些我们...
    晓茜插画阅读 644评论 0 0
  • 测试的时候可以看看你的咪咪好大了吗老婆我好想你我好想你好想你们都在睡觉吗哦你现在是不是觉得
    香香猪_3862阅读 36评论 0 0