Android Retrofit使用指南

TristaLomax 8年前
   <p><img src="https://simg.open-open.com/show/a1126679cf313518dc31e0cb9b088acc.jpg" alt="Android Retrofit使用指南" width="846" height="256"></p>    <p>Retrofit简介</p>    <p><code>Retrofit</code>是大名鼎鼎的 <em>Square</em> 公司开源的适用于<code>Android</code>与<code>Java</code>的网络请求库,官方的介绍非常简短</p>    <pre>  <code class="language-java">A type-safe HTTP client for Android and Java</code></pre>    <p><code>Retrofit</code>使用注解,能够极大的简化网络请求,在2.0版本默认使用<em>Square</em>自家的<code>OkHttp</code>作为底层<code>Http Client</code>,关于如何使用<code>OkHttp</code>配合<code>Retrofit</code>本文后面也会讲到。首先在<code>build.gradle</code>中加入</p>    <pre>  <code class="language-java">  compile 'com.squareup.retrofit2:retrofit:2.0.2'</code></pre>    <p>定义一个网络请求:</p>    <pre>  <code class="language-java">public interface ZhiHuApi {  @GET("users")  Call<List<Repo>> listRepos(@Path("user") String user);  }</code></pre>    <p><code>Retrofit</code>将网络请求转变成了<code>Java interface</code>的形式,<code>interface</code>要获得实例调用<code>listRepos(String user)</code>,需要借助<code>Retrofit.java</code>这个类,通过<code>Retrofit.Builder</code>来配置<code>Retrofit</code>,再通过<code>retrofit.create(final Class<T> service)</code>获取接口的实例</p>    <pre>  <code class="language-java">class Factory {      public static ZhiHuApi create() {      Retrofit retrofit = new Retrofit.Builder()                      .baseUrl("http://news-at.zhihu.com/")                      .build();              return retrofit.create(ZhiHuApi.class);          }  }</code></pre>    <p><code>@GET</code>注解表示请求方式为<code>GET</code>,其中的字符串表示相对<code>URL</code>,而<code>scheme host port</code>被认为是<code>BaseUrl</code>设置到<code>Retrofit</code>中,注意<code>BaseUrl</code>需要以<code>/</code>结尾,这样每个接口定义的相对URL就不需要以<code>/</code>开始。如果在接口中定义的<code>URL</code>为全路径,将用这个全路径作为请求<code>URL</code>,<code>BaseUrl</code>将不起作用。<code>@Path</code>标识get请求的请求参数,上面的<code>listRepos</code>可以认为是在请求<code>http://news-at.zhihu.com/users?user=user</code><br> <code>@POST</code>注解表示请求方式为<code>POST</code>,通常和<code>@FormUrlEncoded</code>注解一起使用,在使用<code>@FormUrlEncoded</code>时可以使用<code>@Field</code>标识表单字段</p>    <pre>  <code class="language-java">@FormUrlEncoded  @POST("user/login.do")  Call<User> login(@Field("username") String userName, @Field("password") String password);</code></pre>    <p>或者使用<code>@FieldMap</code>提交整个<code>map</code></p>    <pre>  <code class="language-java">@FormUrlEncoded  @POST("user/login.do")  Call<User> login(@FieldMap Map<String, String> formMap);</code></pre>    <p>当然你也可以把整个表单封装为一个实体,使用<code>@Body</code>一次提交</p>    <pre>  <code class="language-java">@FormUrlEncoded  @POST("user/login.do")  Call<User> login(@Body User user);</code></pre>    <p><code>Multipart</code>请求时使用<code>@Multipart</code>注解,用<code>@Part</code>标识每个<code>RequestBody</code></p>    <pre>  <code class="language-java">@Multipart  @PUT("user/photo")  Call<User> updateUser(@Part("photo") RequestBody photo, @Part("description") RequestBody description);</code></pre>    <p>定义好请求之后就可以调用<code>call.enqueue()</code>来执行请求,需要传入<code>Callback<T></code>,其中<code>T</code>的类型编译器会根据<code>Call<T></code>中的类型来判断,<code>Retrofit</code>和其他网络请求库一样对于<em>Android</em>平台做了线程切换,请求在后台执行,<code>Callback<T></code>会回到<code>main (UI) thread</code>,如果是<code>Java</code>程序<code>Callback<T></code>会继续回到调用它的线程。</p>    <pre>  <code class="language-java">ZhiHuApi zhiHuApi = BaseNetwork.Factory.create(ZhiHuApi.class);          Call<User> call = zhiHuApi.login("username", "pwd");          call.enqueue(new Callback<User>() {              @Override              public void onResponse(Call<User> call, Response<User> response) {                }                @Override              public void onFailure(Call<User> call, Throwable t) {                }          });</code></pre>    <p>为请求添加请求头时使用<code>@Headers</code>,这里就不做举例,因为app中通常是每个请求都需要携带请求头,不建议在<code>Retrofit</code>定义请求时传入,而是使用<code>OkHttp</code>来实现统一请求头。</p>    <h2>Converter</h2>    <p><code>Retrofit</code>在默认情况下只能将<code>Http</code>的响应体反序列化到<code>OkHttp</code>的<code>ResponseBody</code>中,加入<code>Converter</code>可以将返回的数据直接格式化成你需要的样子,现有6个<code>Converter</code>可以直接使用:</p>    <ul>     <li>Gson: com.squareup.retrofit2:converter-gson</li>     <li>Jackson: com.squareup.retrofit2:converter-jackson</li>     <li>Moshi: com.squareup.retrofit2:converter-moshi</li>     <li>Protobuf: com.squareup.retrofit2:converter-protobuf</li>     <li>Wire: com.squareup.retrofit2:converter-wire</li>     <li>Simple XML: com.squareup.retrofit2:converter-simplexml</li>     <li>Scalars (primitives, boxed, and String): com.squareup.retrofit2:converter-scalars</li>    </ul>    <p>之后在代码里加入(此处以<code>GsonConverterFactory</code>为例)</p>    <pre>  <code class="language-java">  Retrofit retrofit =new Retrofit.Builder()                      .baseUrl(Constants.BASE_HTTP_URL)                      .addConverterFactory(GsonConverterFactory.create())                      .build();</code></pre>    <p>返回的数据会使用<code>Gson</code>解析为对应传入的实体类,你也可以自定义<code>Converter</code>来实现更复杂的需求,只需要<code>extends Converter.Factory</code>然后重写</p>    <pre>  <code class="language-java">  @Override    public Converter<ResponseBody, ?> responseBodyConverter(Type type, Annotation[] annotations,        Retrofit retrofit) {          //your own implements    }      @Override    public Converter<?, RequestBody> requestBodyConverter(Type type,       Annotation[] parameterAnnotations, Annotation[] methodAnnotations, Retrofit retrofit) {         //your own implements    }</code></pre>    <p><code>Retrofit</code>终归只是应用层的<code>api</code>,真正的执行器是<code>OkHttp</code>,较为复杂的需求都需要从执行层入手,可以做到<code>Retrofit</code>对外不变的多种自定义统一封装。</p>    <h2>OkHttp配合Retrofit使用</h2>    <p>前文已经提到在<code>Retrofit 2.0</code>中已经默认使用<code>OkHttp</code>作为网络请求执行器,关于<code>OkHttp</code>的优点简单提一下:(<a href="/misc/goto?guid=4959673778567258052">原文链接</a>)</p>    <ul>     <li>1.支持HTTP2/SPDY黑科技</li>     <li>2.socket自动选择最好路线,并支持自动重连</li>     <li>3.拥有自动维护的socket连接池,减少握手次数</li>     <li>4.拥有队列线程池,轻松写并发</li>     <li>5.拥有Interceptors轻松处理请求与响应(比如透明GZIP压缩,LOGGING)</li>     <li>6.基于Headers的缓存策略</li>    </ul>    <p>想要使用<code>OkHttp</code>为<code>Retrofit</code>提供更高的定制性,给<code>Retrofit</code>设置自定义的<code>OkHttpClient</code>就可以了</p>    <pre>  <code class="language-java">Retrofit retrofit = new Retrofit.Builder()                    .baseUrl(Constants.BASE_HTTP_URL)                    .client(client)                    .build();</code></pre>    <p>之后就是构建一个<code>OkHttpClient</code></p>    <pre>  <code class="language-java">OkHttpClient client = new OkHttpClient.Builder()             // 向Request Header添加一些业务相关数据,如APP版本,token             .addInterceptor(new HeadInterceptor())             //日志Interceptor,可以打印日志             .addInterceptor(logging)             // 连接超时时间设置             .connectTimeout(10, TimeUnit.SECONDS)             // 读取超时时间设置             .readTimeout(10, TimeUnit.SECONDS)             // 失败重试             .retryOnConnectionFailure(true)             // 支持Https需要加入SSLSocketFactory             .sslSocketFactory(sslSocketFactory)             // 信任的主机名,返回true表示信任,可以根据主机名和SSLSession判断主机是否信任             .hostnameVerifier(new HostnameVerifier() {                 @Override                 public boolean verify(String hostname, SSLSession session) {                     return true;                 }             })             // 使用host name作为cookie保存的key             .cookieJar(new CookieJar() {                 private final HashMap<HttpUrl, List<Cookie>> cookieStore = new HashMap<>();                   @Override                 public void saveFromResponse(HttpUrl url, List<Cookie> cookies) {                     cookieStore.put(HttpUrl.parse(url.host()), cookies);                 }                   @Override                 public List<Cookie> loadForRequest(HttpUrl url) {                     List<Cookie> cookies = cookieStore.get(HttpUrl.parse(url.host()));                     return cookies != null ? cookies : new ArrayList<Cookie>();                 }             })             .build();</code></pre>    <p>如果设置了<code>sslSocketFactory</code>却没有配置对应的<code>hostnameVerifier</code>,那么Https请求是无法成功的。上面用到两个<code>Interceptor</code>分别是<code>HeadInterceptor</code>和<code>HttpLoggingInterceptor</code>,分别是用来添加请求头和打印请求日志的拦截器,<code>OkHttp</code>支持自定义拦截器,例如下面代码自定义的<code>HeadInterceptor</code>为请求加入<code>Headers</code></p>    <pre>  <code class="language-java">public class HeadInterceptor implements Interceptor {        @Override      public Response intercept(Chain chain) throws IOException {          Request originalRequest = chain.request();          Request compressedRequest = originalRequest.newBuilder()                  .headers(Headers.of(getHeaders()))                  .build();          return chain.proceed(compressedRequest);      }  }</code></pre>    <p>有时服务器会对<code>POST</code>提交的表单做参数校验,一种方式是在请求头里加入特定方式加密过的表单参数的<code>Map</code>,那么就需要先获取到请求的<code>Map</code>,通过<code>FormBody</code>可以实现</p>    <pre>  <code class="language-java">// if the server needs to verify post params, use this to get post params;    RequestBody oidBody = originalRequest.body();    Map<String, String> params = new HashMap<>();    if (oidBody instanceof FormBody) {      FormBody formBody = (FormBody) oidBody;      for (int i = 0; i < formBody.size(); i++) {          params.put(formBody.encodedName(i), formBody.encodedValue(i));      }    }</code></pre>    <p><code>HttpLoggingInterceptor</code>是_Square_ 提供的请求信息日志打印工具类,如果需要可以在<code>build.gradle</code>中加入</p>    <pre>  <code class="language-java">  compile 'com.squareup.okhttp3:logging-interceptor:3.2.0'</code></pre>    <p>可以根据不同情况配置日志输出的<code>Level</code>:</p>    <ul>     <li><strong>NONE</strong> 不输出日志</li>     <li><strong>BASIC</strong> 只输出请求方式响应码等基本信息</li>     <li><strong>HEADERS</strong> 只输出请求和响应的头部信息</li>     <li><strong>BODY</strong> 输出请求和响应的头部和请求体信息</li>    </ul>    <p>另外如果遇到两个接口有相互依赖关系,必须请求完第一个接口拿到数据后才知道第二个请求的<code>URL</code>,通常我们会定义两个<code>Retrofit</code>,因为<code>Retrofit</code>的<code>BaseUrl</code>的统一配置的,不过现在可以通过实现动态<code>BaseUrl</code>来避免这个问题,先看<code>DynamicBaseUrlInterceptor</code>的代码</p>    <pre>  <code class="language-java">public class DynamicBaseUrlInterceptor implements Interceptor {      private volatile String host;        public void setHost(String host) {          this.host = host;      }        @Override      public Response intercept(Chain chain) throws IOException {          Request originalRequest = chain.request();          if (!TextUtils.isEmpty(host)) {              HttpUrl newUrl = originalRequest.url().newBuilder()                      .host(host)                      .build();              originalRequest = originalRequest.newBuilder()                      .url(newUrl)                      .build();          }            return chain.proceed(originalRequest);      }  }</code></pre>    <p>在<code>BaseUrl</code>改变时只需要<code>setHost()</code>就可以让下次请求的<code>Baseurl</code>改变</p>    <h2>Retrofit 与 RxJava 结合使用</h2>    <p>本节需要对<code>RxJava</code>基本用法有了解,如果不了解可以忽略或者先去熟悉一下<code>RxJava</code>的<a href="/misc/goto?guid=4959631832560174430">wiki</a>,介绍的目的是因为两者结合使用确实很方便,关于<code>RxJava</code>之后会单独写。</p>    <p><code>RxJava</code>是<code>Rx</code>(全称<code>Reactive Extensions</code>)家族中的一员,是最近很火的响应式编程库,官方对于它的解释很简单</p>    <pre>  <code class="language-java">RxJava – Reactive Extensions for the JVM – a library for composing asynchronous and event-based programs using observable sequences for the Java VM.</code></pre>    <p>一个异步的基于事件的观察者序列,可以理解为扩展的观察者模式,在Android中使用<code>RxJava</code>需要引入两个<code>compile</code>,<code>RxAndroid</code>是专为Android平台打造来提供主线程切换等便利的工具项目。</p>    <pre>  <code class="language-java">   compile 'io.reactivex:rxandroid:1.2.0'     // Because RxAndroid releases are few and far between, it is recommended you also     // explicitly depend on RxJava's latest version for bug fixes and new features.     compile 'io.reactivex:rxjava:1.1.5'</code></pre>    <p><code>Retrofit</code>提供了<code>CallAdapterFactory</code>,它是一个知道如何将<code>call</code>实例转换成其他类型的工厂类,目前支持的有:</p>    <ul>     <li>RxJava</li>     <li>Guava</li>     <li>Java8</li>    </ul>    <p>这些和<code>Retrofit</code>本身都是分离的,需要单独引入<code>compile</code>例如</p>    <pre>  <code class="language-java">compile 'com.squareup.retrofit2:adapter-rxjava:2.0.0'</code></pre>    <p>在代码中配置<code>CallAdapterFactory</code></p>    <pre>  <code class="language-java">Retrofit retrofit =new Retrofit.Builder()                     .baseUrl(Constants.BASE_HTTP_URL)                     .addConverterFactory(GsonConverterFactory.create())                     .addCallAdapterFactory(RxJavaCallAdapterFactory.create())                     .build();</code></pre>    <p>之后就可以把请求的返回改为<code>Observable<T></code>了</p>    <pre>  <code class="language-java">@GET("users")  Observable<List<Repo>> listRepos(@Path("user") String user);</code></pre>    <p>请求时只需要</p>    <pre>  <code class="language-java">BaseNetwork.Factory.create(Foo.class)                  .listRepos("user")                  .observeOn(AndroidSchedulers.mainThread())//观察者所在的线程                  .subscribeOn(Schedulers.io())//请求执行的线程                  //如果正常执行会顺序调用onNext,onCompleted,如果出错则会调用onError                  .subscribe(new Observer<List<Repo>>() {                      @Override                      public void onCompleted() {                        }                        @Override                      public void onError(Throwable e) {                        }                        @Override                      public void onNext(List<Repo> list) {                        }                  });</code></pre>    <p>如果项需要服务器返回固定的格式用来定义一些业务上的错误如下</p>    <pre>  <code class="language-java">{    "state":0,//状态码,0为业务正常    "msg":"",//如果业务出错,携带错误信息    "data":{}//包含实际业务实体  }</code></pre>    <p>需要定义统一的响应实体,根据<code>T</code>传入的类型来获取业务实体真实的类型</p>    <pre>  <code class="language-java">public class BaseResult<T> {      private int state;      private String msg;      private T data;        public int getState() {          return state;      }        public void setState(int state) {          this.state = state;      }        public String getMsg() {          return msg;      }        public void setMsg(String msg) {          this.msg = msg;      }        public T getData() {          return data;      }        public void setData(T data) {          this.data = data;      }  }</code></pre>    <p>请求中的泛型类型需要是<code>BaseResult<T></code></p>    <pre>  <code class="language-java">@GET("users")  Observable<BaseResult<List<Repo>>> listRepos(@Path("user") String user);</code></pre>    <p>调用时也会有改变,需要经过一次拆解统一返回,处理错误的过程</p>    <pre>  <code class="language-java">BaseNetwork.Factory.create(Foo.class)                  .listRepos("user")                  .flatMap(new NetworkResultFunc1<List<Repo>>())                  .observeOn(AndroidSchedulers.mainThread())                  .subscribeOn(Schedulers.io())                  .subscribe(observer);</code></pre>    <p><code>flatMap</code>需要传入<code>Func1<T, R></code>,<code>Func1<T, R></code>继承了<code>Function</code>,只有一个方法,将泛型参数列表的第一个转换为第二个返回,它可以将<code>Observable</code>做一个展开,并返回一个新的<code>Observable</code></p>    <pre>  <code class="language-java">public interface Func1<T, R> extends Function {      R call(T t);  }</code></pre>    <p><code>NetworkResultFunc1<List<Repo>></code>实现了<code>Func1<T, R></code>,代码如下</p>    <pre>  <code class="language-java">public class NetworkResultFunc1<T> implements Func1<BaseResult<T>, Observable<T>> {        @Override      public Observable<T> call(final BaseResult<T> tBaseResult) {          return Observable.create(new Observable.OnSubscribe<T>() {              @Override              public void call(Subscriber<? super T> subscriber) {                  int state = tBaseResult.getState();                  String msg = tBaseResult.getMsg();                  switch (state) {                      case 0://if success, return data to client                          subscriber.onNext(tBaseResult.getData());                          break;                      case 1000://if this means error                          subscriber.onError(new ApiException(state, msg));                          break;                  }                  subscriber.onCompleted();//no error, will execute onCompleted()              }          });      }  }</code></pre>    <p>如果state为0,则调用<code>subscriber.onNext()</code>向调用者返回数据,当state不等于0时意味着业务出错了,向<code>subscriber.onError()</code>中抛了一个<code>ApiException</code>,这样在<code>Observer</code>处会回调<code>onError()</code>终止整个事件流,调用者也能获得业务错误的相关信息。<code>ApiException</code>代码如下,就是一个自定义的<code>RuntimeException</code></p>    <pre>  <code class="language-java">public class ApiException extends RuntimeException {      private int state;      private String msg;      public ApiException(int state, String msg) {          this.state = state;          this.msg = msg;      }        public int getState() {          return state;      }        public String getMsg() {          return msg;      }  }</code></pre>    <p>对于<code>Retrofit</code>的介绍就先到这里,相信看到这里,你已经能够在项目中优雅的使用<code>Retrofit</code>了。</p>    <p><br> 文/<a href="/misc/goto?guid=4959673778685339668">winter1991</a>(简书)<br>  </p>    <p> </p>