2024年Android最新Retrofit源码学习五:Retrofit中同步、异步请求解析(2),2024年最新面试必备物品

最后

代码真的是重质不重量,质量高的代码,是当前代码界提倡的,当然写出高质量的代码肯定需要一个相当高的专业素养,这需要在日常的代码书写中逐渐去吸收掌握,谁不是每天都在学习呀,目的还不是为了一个,为实现某个功能写出高质量的代码。

所以,长征路还长,大家还是好好地做个务实的程序员吧。

最后,小编这里有一系列Android提升学习资料,有兴趣的小伙伴们可以来看下哦~

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

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

Retrofit源码学习四:Retrofit源码详解二

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

Retrofit源码学习六:框架中的设计模式

Retrofit同步请求和异步请求的区别:其实,这两个方法大体相同。唯一的不同在,异步请求会将回调方法交给回调执行器Executor,然后由Executor指定不同的线程去完成不同的操作。

我们先来总结一下同步方法的步骤:

同步请求:OkHttpCall.execute()

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

一、用ParameterHandler对网络请求接口中的方法、参数进行解析

二、用ServiceMethod创建一个Okhttp的Request对象

三、通过okhttp发送网络请求

四、用数据解析器converter解析服务器返回的数据(Retrofit中默认用的是GsonConverterFactory)

如下图所示:

我们看具体的代码:

//class OkHttpCall开始

final class OkHttpCall implements Call {

private okhttp3.Call rawCall;

@Override

public Response execute() throws IOException {

okhttp3.Call call;

synchronized (this) {

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

executed = true;

if (creationFailure != null) {

if (creationFailure instanceof IOException) {

throw (IOException) creationFailure;

} else {

throw (RuntimeException) creationFailure;

}

}

call = rawCall;

if (call == null) {

try {

call = rawCall = createRawCall();

} catch (IOException | RuntimeException e) {

creationFailure = e;

throw e;

}

}

}

if (canceled) {

call.cancel();

}

return parseResponse(call.execute());

}

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 parseResponse(okhttp3.Response rawResponse) throws IOException {

ResponseBody rawBody = rawResponse.body();

// Remove the body’s source (the only stateful object) so we can pass the response

// along.

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)

总结

这次面试问的还是还是有难度的,要求当场写代码并且运行,也是很考察面试者写代码
因为Android知识体系比较庞大和复杂的,涉及到计算机知识领域的方方面面。在这里我和身边一些朋友特意整理了一份快速进阶为Android高级工程师的系统且全面的学习资料。涵盖了Android初级——Android高级架构师进阶必备的一些学习技能。

附上:我们之前因为秋招收集的二十套一二线互联网公司Android面试真题(含BAT、小米、华为、美团、滴滴)和我自己整理Android复习笔记(包含Android基础知识点、Android扩展知识点、Android源码解析、设计模式汇总、Gradle知识点、常见算法题汇总。)

里面包含不同方向的自学编程路线、面试题集合/面经、及系列技术文章等,资源持续更新中…

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

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

id扩展知识点、Android源码解析、设计模式汇总、Gradle知识点、常见算法题汇总。)
[外链图片转存中…(img-11D6KI8h-1714971687682)]
里面包含不同方向的自学编程路线、面试题集合/面经、及系列技术文章等,资源持续更新中…

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值