Retrofit分析之框架设计艺术

   2016-10-07 0
核心提示:Retrofit使用者会觉得接口+注解方式去写网络请求很吊。但当你真正的去看它的源码,会被它独特,漂亮的解耦方式所吸引,整个结构运用了动态代理,策略模式,Builder模式,工厂等设计模式。Retrofit v2.1基于Okhttp3,可以说是对okhttp进行二次封装。先感受一下

Retrofit使用者会觉得接口+注解方式去写网络请求很吊。但当你真正的去看它的源码,

会被它独特,漂亮的解耦方式所吸引,整个结构运用了动态代理,策略模式,Builder模式,工厂等设计模式。

Retrofit v2.1基于Okhttp3,可以说是对okhttp进行二次封装。

先感受一下使用方式。。。。

Retrofit官方使用方式

//定义HTTP API接口
public interface LocationService {
@Headers({
"Accept: application/json",
"Content-Type: application/json"
})
@GET("geo/ip?")
Call<Location> currentLocation();
}

Retrofit retrofit = new Retrofit.Builder()
.baseUrl(baseUrl)
.addConverterFactory(GsonConverterFactory.create())
.build();
LocationService service = retrofit.create(LocationService.class);

retrofit2.Call<Location> call = service.currentLocation();
call.enqueue(new retrofit2.Callback<Location>() {
@Override
public void onResponse(retrofit2.Call<Location> call, retrofit2.Response<Location> response) {
Log.d(TAG, "onResponse: " + response.body());
}

@Override
public void onFailure(retrofit2.Call<Location> call, Throwable t) {
Log.d(TAG, "onFailure: " + t.getMessage());
}
});

Retrofit + RxJava 的使用方式

public interface RxLocationService {
@Headers({
"Accept: application/json",
"Content-Type: application/json"
})
@GET("geo/ip?)
Observable<Location> currentLocation();
}

Retrofit retrofit = new Retrofit.Builder()
.baseUrl(baseUrl)
.addCallAdapterFactory(RxJavaCallAdapterFactory.create())
.addConverterFactory(GsonConverterFactory.create())
.build();
RxLocationService service = retrofit.create(RxLocationService.class);

Observable<Location> ob = service.currentLocation();
ob.subscribeOn(Schedulers.newThread())
.subscribeOn(AndroidSchedulers.mainThread())
.subscribe(new Subscriber<Location>() {
@Override
public void onCompleted() {
Log.d(TAG, "onCompleted: ");
}

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

@Override
public void onNext(Location location) {
Log.d(TAG, "onNext: " + location);
}
});

Retrofit的配置

Retrofit retrofit = new Retrofit.Builder()
.baseUrl(baseUrl)
.callFactory(new OkHttpClient())
.addCallAdapterFactory(RxJavaCallAdapterFactory.create())
.addConverterFactory(GsonConverterFactory.create())
.build();

通过addCallAdapterFactory和addConverterFactory配置不同的OkHttpClient,

CallAdapter(调用方式的适配器),Converter(数据转换器)。

下面时本人分析的流程图:

Retrofit分析之框架设计艺术

CallAdapter 接口

这个接口决定HTTP API接口里每个方法的返回值类型。

Retrofit的create方法

return (T) Proxy.newProxyInstance(service.getClassLoader(), new Class<?>[] { service },
new InvocationHandler() {
private final Platform platform = Platform.get();

@Override public Object invoke(Object proxy, Method method, Object... args)
throws Throwable {
// If the method is a method from Object then defer to normal invocation.
if (method.getDeclaringClass() == Object.class) {
return method.invoke(this, args);
}
if (platform.isDefaultMethod(method)) { // java8的默认方法
return platform.invokeDefaultMethod(method, service, proxy, args);
}
ServiceMethod<Object, Object> serviceMethod =
(ServiceMethod<Object, Object>) loadServiceMethod(method);
OkHttpCall<Object> okHttpCall = new OkHttpCall<>(serviceMethod, args);
return serviceMethod.callAdapter.adapt(okHttpCall);
}
});

运用了动态代理生成API接口的实现,动态代理会有个invoke方法,当调用接口的一个方法时会被调用。

例如:上面使用方式提到的LocationService,当调用locationService.currentLocation()时,invoke方法

会被调用,返回值就是Call

ServiceMethod:存储了API接口的需要的注解,包含request mothod, request header等信息。

loadServiceMethod(method)整个方法解析当前方法的注解等属性。

OkHttpCall对okhttp进行了封装。传入ServiceMethod,args进行OkHttpCall实例化,

最终调用serviceMethod.callAdapter.adapt(okHttpCall) 产生CallAdapter.

public interface CallAdapter<R, T>{}
Type responseType();
<R> T adapt(Call<R> call);

abstract class Factory {

public abstract CallAdapter<?> get(Type returnType, Annotation[] annotations,
Retrofit retrofit);

protected static Type getParameterUpperBound(int index, ParameterizedType type) {
return Utils.getParameterUpperBound(index, type);
}

protected static Class<?> getRawType(Type type) {
return Utils.getRawType(type);
}
}
}

CallAdapter的作用:

1.当调用API接口一个方法时就会产生一个CallAdapter实现类;

2.adapt方法返回的T决定API接口方法的返回类型;(对应例子currentLocation返回值Call类型)

3.responseType方法返回的Type时网络请求返回类型。(对应Call上面泛型Location的类型)

在Retrofit的配置里讲到,可以配置多种CallAdapter,就可以实现API接口返回不同的类型,从而可以又不同调用方式。

也就产生上面提到的 Retrofit官方使用方式 和 Retrofit + RxJava 的使用方式。

官方使用方式的实现在ExecutorCallAdapterFactory.java, ExecutorCallAdapterFactory是在Retrofit的build方法配置了默认的CallAdapterFactory。

//在ExecutorCallAdapterFactory.java
return new CallAdapter<Object, Call<?>>() {
@Override public Type responseType() {
return responseType;
}

@Override public Call<Object> adapt(Call<Object> call) {
return new ExecutorCallbackCall<>(callbackExecutor, call);
}
};

//adapt的方法new ExecutorCallbackCall<>(callbackExecutor, call)

//在ExecutorCallbackCall.java
// 上面官方使用方式例子里调用如下:
// retrofit2.Call<Location> call = service.currentLocation();
// call.enqueue(callback);

@Override public void enqueue(final Callback<T> callback) {
if (callback == null) throw new NullPointerException("callback == null");

delegate.enqueue(new Callback<T>() { //delegate 实质是OkHttpCalls
@Override public void onResponse(Call<T> call, final Response<T> response) {
callbackExecutor.execute(new Runnable() {
@Override public void run() {
if (delegate.isCanceled()) {
// Emulate OkHttp's behavior of throwing/delivering an IOException on cancellation.
callback.onFailure(ExecutorCallbackCall.this, new IOException("Canceled"));
} else {
callback.onResponse(ExecutorCallbackCall.this, response);
}
}
});
}
}
...
}

OkHttpCall:
1. Response<T> execute() // 同步
2. void enqueue(final Callback<T> callback) // 异步

```sh
// OkHttpCall的关键代码
private okhttp3.Call createRawCall() throws IOException {
Request request = serviceMethod.toRequest(args);
okhttp3.Call call = serviceMethod.callFactory.newCall(request);
if (call == null) {
throw new NullPointerException("Call.Factory returned null.");
}
return call;
}

Response<T> parseResponse(okhttp3.Response rawResponse) throws IOException {
...
try {
T body = serviceMethod.toResponse(catchingBody);
return Response.success(body, rawResponse);
} catch (RuntimeException e) {
...
}


//ServiceMethod的toResponse方法实现
R toResponse(ResponseBody body) throws IOException {
return responseConverter.convert(body);
}
```
通过ServiceMethod的toRequest(args)方法转成okhttp的Request, 然后通过okhtpp去完成网络请求。
返回结果通过toResponse(catchingBody)交给Converter,最终转成T调回。

Retrofit + RxJava 的使用方式实现在RxJavaCallAdapterFactory.java。这里不产开表述了。

Converter接口

Retrofit默认配置里没有配置ConverterFactory, 初次使用时通常忘了配置一个ConverterFactory导致出错。

Retrofit以插件的形式提供了多种Converter实现:

  • Gson: com.squareup.retrofit2:converter-gson
  • Jackson: com.squareup.retrofit2:converter-jackson
  • Moshi: com.squareup.retrofit2:converter-moshi
  • Protobuf: com.squareup.retrofit2:converter-protobuf
  • Wire: com.squareup.retrofit2:converter-wire
  • Simple XML: com.squareup.retrofit2:converter-simplexml
  • Scalars (primitives, boxed, and String): com.squareup.retrofit2:converter-scalars
 
标签: Retrofit
反对 0举报 0 评论 0
 

免责声明:本文仅代表作者个人观点,与乐学笔记(本网)无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
    本网站有部分内容均转载自其它媒体,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责,若因作品内容、知识产权、版权和其他问题,请及时提供相关证明等材料并与我们留言联系,本网站将在规定时间内给予删除等相关处理.

  • [Java] Retrofit2.0 如何进行GBK编码
    对Retrofit + OkHttp还不熟悉的人可以点传送门,先看下这两个东西的使用。Retrofit:https://github.com/square/retrofitOkHttp:https://github.com/square/okhttp分析接口文档要求Post请求,字段使用GBK编码我们先按照Retrofit的规范和接口文档来写接口: @PO
    12-23 RetrofitJava
  • Android常用的开源项目及其比较系列-Retrofit进
    上一篇我们谈了谈Androiod开源项目的网络框架, 比较了它们之间的优缺点,原文在这里。今天我们着重谈谈Retrofit框架如何更友好的使用,本着提出问题解决问题的原则,也为大家以后解决问题提供基本思路。目前都有哪些问题?根据官方Demo, 简单使用是这么样的
  • Android MVP架构实践
    Android MVP架构实践
    首先声明一下,没有完美的架构,只要适合自己的项目,那就是最好的架构。本例子是MVP + Retrofit + RxJava结合的例子,但本文的重点在于讲解MVP架构,所以涉及Retrofit和RxJava的部分将直接略过,默认读者已了解这两部分内容,如有需要,请自行查阅相关资料,
  • 从 Retrofit 源码学习 Java 的动态代理的使用
    Retrofit 是当前 Android 最流行的 HTTP 网络库之一了,其使用方式比较特殊,是通过定义一个接口类,通过给接口中方法和方法参数添加注解的方式来定义网络请求接口。这种风格下定义一个网络接口变得很简单。不过 Retrofit 是如何使用一个接口的 Class 创建出
    11-13 RetrofitJava
  • 打造企业级网络请求框架集合 retrofit+gson+mvp
    打造企业级网络请求框架集合 retrofit+gson+m
    本文是企业级网络框架第二篇主要讲MVP模式和Gson在Retrofit网络请求框架下的使用方式。(已更新为一篇) 对MVP不了解的请看梦之鬼索MVP模式在Android中的设计和实现 http://blog.csdn.net/androidmsky/article/details/52248797 对Retrofit还不了解的情看 打
    10-31 RetrofitGson
  • Android Retrofit框架解析
    随着Google对HttpClient的摒弃,和Volley的逐渐没落,OkHttp开始异军突起,而Retrofit则对okHttp进行了强制依赖。Retrofit也是Square公司开发的一款针对Android网络请求的框架,其实质就是对okHttp的封装,使用面向接口的方式进行网络请求,利用动态生成的代理类
    10-04 Retrofit
  • Android网络开源库-Retrofit(五)简易封装
    1.前言Rrtrofit的扩展性很强,如果对retrofit不熟悉的话,是很难应对各种各样的需求的。因此,在这里,做一下简单的封装。主要为了下面三点需求:使用简单加密处理错误处理2.怎样才能简单使用为了简单粗暴,我做了以下工作。使用单例Retrofit引入RxJava在这里
  • Android网络开源库-Retrofit(四)文件相关
    以前写过一些retrofit的相关文章,当时只是自己学习研究的,最近项目,加入了retrofit,因此遇到了一些问题,需要记录一下。1.前言在以前,写过retrofit上传文件相关,但是,需求总是变化的。前面的,介绍了上传进度的监听,但是,那时候是监听单文件进度。虽
  • 急速开发系列——Retrofit 响应数据及异常处理
    今天我们来谈谈客户端对通讯协议的处理,主要分为三部分:约定响应数据格式,响应数据的自动映射以及错误处理三部分。由于数据协议采用json的居多,因此我们在此基础上进行说明。约定响应数据格式协议格式通常来说,你拿到的设计文档中会存在通信协议的说明,
    09-29 RetrofitGson
  • 使用Android API最佳实践
    使用Android API最佳实践
    写在前面 现在,Android应用程序中集成第三方API已十分流行。应用程序都有自己的网络操作和缓存处理机制,但是大部分比较脆弱,没有针对网络糟糕情况进行优化。感谢 Square lnc这家有创新精神的公司,将信用卡商业交易带到手机上。现在有了一系列高质量开源库
    09-20 APIRetrofit
点击排行