Retrofit源码学习五:Retrofit中同步、异步请求解析

rawResponse = rawResponse.newBuilder()

.body(new NoContentResponseBody(rawBody.contentType(), rawBody.contentLength()))

.build();

int code = rawResponse.code();

if (code < 200 || code >= 300) {

try {

// Buffer the entire body to avoid future I/O.

ResponseBody bufferedBody = Utils.buffer(rawBody);

return Response.error(bufferedBody, rawResponse);

} finally {

rawBody.close();

}

}

if (code == 204 || code == 205) {

return Response.success(null, rawResponse);

}

ExceptionCatchingRequestBody catchingBody = new ExceptionCatchingRequestBody(rawBody);

try {

T body = serviceMethod.toResponse(catchingBody);

return Response.success(body, rawResponse);

} catch (RuntimeException e) {

// If the underlying source threw an exception, propagate that rather than indicating it was

// a runtime exception.

catchingBody.throwIfCaught();

throw e;

}

}

}

// class OkHttpCall结束

//class ServiceMethod开始

final class ServiceMethod {

Request toRequest(Object… args) throws IOException {

RequestBuilder requestBuilder = new RequestBuilder(httpMethod, baseUrl, relativeUrl, headers,

contentType, hasBody, isFormEncoded, isMultipart);

@SuppressWarnings(“unchecked”)

// It is an error to invoke a method with the wrong arg types.

ParameterHandler[] handlers = (ParameterHandler[]) parameterHandlers;

int argumentCount = args != null ? args.length : 0;

if (argumentCount != handlers.length) {

throw new IllegalArgumentException(“Argument count (” + argumentCount

  • “) doesn’t match expected count (” + handlers.length + “)”);

}

for (int p = 0; p < argumentCount; p++) {

handlers[p].apply(requestBuilder, args[p]);

}

return requestBuilder.build();

}

T toResponse(ResponseBody body) throws IOException {

return responseConverter.convert(body);

}

}

//class ServiceMethod结束

其实,从第4行和第8行我们已经看到了,OkHttpCall.execute()方法中调用Call的其实就是okhttp3的Call。

在第25行中,call = rawCall = createRawCall(); 我们看到调用了createRawCall()方法给call和rawCall赋值。

我们跟进去看一下,这个方法就是39行的方法:在40行和41行,类似于okhttp的创建request和call的请求。

这个印证了我们的第二个步骤。

这里需要我们重点关注的是41行的toRequest(args)方法,我们跟踪这个方法:就是90行的这个方法。

在96行,我们看到了ParameterHandler,这就印证了我们总结的第一个步骤。

在创建好request请求和call对象后,我们接着往下走,看37行:return parseResponse(call.execute())

call.execute()其实就是okhttp的同步方法execute(),

而parseResponse,根据名称就知道这个方法是用来解析服务器返回的数据的,这印证了我们的第四个步骤。

跟踪这个方法,从49行到83行:

在58行,我们先获取到相应码

在63行,看到了回调的Response.error(bufferedBody, rawResponse)

在70和76行,看到了回调的Response.success(body, rawResponse)

这些不必多讲,我们主要看75行的 T body = serviceMethod.toResponse(catchingBody);

跟踪这个方法,我们看到

T toResponse(ResponseBody body) throws IOException {

return responseConverter.convert(body);

}

它其实就是调用了转换器来转换数据。

总结:


Retrofit做的就是通过接口和注解把http请求进行了包装。因此,当我们在使用Retrofit的时候,我们只需要将焦点放在接口的创建上,通过接口来配置方法和参数,其他的工作都由Retrofit的内部来完成。而它的内部原理也很简单:就是通过动态代理将我们客户端写好的接口中的方法转化成ServiceMethod对象,我们就通过这个ServiceMethod对象获取我们需要的信息,而最终的网络请求通过底层的okhttp库来完成。

异步请求:OkHttpCall.enqueue()

=========================

final class OkHttpCall implements Call{

@Override

public void enqueue(final Callback callback) {

if (callback == null) throw new NullPointerException(“callback == null”);

okhttp3.Call call;

Throwable failure;

synchronized (this) {

if (executed) throw new IllegalStateException(“Already executed.”);

executed = true;

call = rawCall;

failure = creationFailure;

if (call == null && failure == null) {

try {

call = rawCall = createRawCall();

} catch (Throwable t) {

failure = creationFailure = t;

}

}

}

if (failure != null) {

callback.onFailure(this, failure);

return;

}

if (canceled) {

call.cancel();

}

call.enqueue(new okhttp3.Callback() {

@Override public void onResponse(okhttp3.Call call, okhttp3.Response rawResponse)

throws IOException {

Response response;

try {

response = parseResponse(rawResponse);

} catch (Throwable e) {

callFailure(e);

return;

}

callSuccess(response);

}

@Override public void onFailure(okhttp3.Call call, IOException e) {

try {

callback.onFailure(OkHttpCall.this, e);

} catch (Throwable t) {

t.printStackTrace();

}

}

private void callFailure(Throwable e) {

try {

callback.onFailure(OkHttpCall.this, e);

} catch (Throwable t) {

t.printStackTrace();

}

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
img

最后

上面这些公司都是时下最受欢迎的互联网大厂,他们的职级、薪资、福利也都讲的差不多了,相信大家都是有梦想和野心的人,心里多少应该都有些想法。

也相信很多人也都在为即将到来的金九银十做准备,也有不少人的目标都是这些公司。

我这边有不少朋友都在这些厂工作,其中也有很多人担任过面试官,上面的资料也差不多都是从朋友那边打探来的。除了上面的信息,我这边还有这些大厂近年来的面试真题及解析,以及一些朋友出于兴趣和热爱一起整理的Android时下热门知识点的学习资料

部分文件:


一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
img
官,上面的资料也差不多都是从朋友那边打探来的。除了上面的信息,我这边还有这些大厂近年来的面试真题及解析,以及一些朋友出于兴趣和热爱一起整理的Android时下热门知识点的学习资料

部分文件:
[外链图片转存中…(img-x7cl9ZiN-1712806205839)]
[外链图片转存中…(img-bvtQlgJw-1712806205839)]
[外链图片转存中…(img-W2OivlKE-1712806205839)]

一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
[外链图片转存中…(img-DGdvLYJJ-1712806205840)]

  • 23
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值