OkHttp3源码详解之Okhttp任务队列工作原理(三)

  1. if (executorService == null) {
  2. executorService = , Integer.MAX_VALUE, , TimeUnit.SECONDS,
  3. new SynchronousQueue(), Util.threadFactory(“OkHttp Dispatcher”, false));
  4. }
  5. return executorService;
  6. }

1.2 Call对象

了解源码或使用过okhttp的都知道。 okttp的操作元是Call对象。异步的实现是RealCall.AsyncCall。而 AsyncCall是实现的一个Runnable接口。

  1. final class AsyncCall extends NamedRunnable {}

所以Call本质就是一个Runable线程操作元肯定是放进excutorService中直接启动的。

2 线程池的复用和管理

2.1 图解

为了完成调度和复用,定义了两个队列分别用作等待队列和执行任务的队列。这两个队列都是Dispatcher 成员变量。Dispatcher是一个控制执行,控制所有Call的分发和任务的调度、通信、清理等操作。这里只介绍异步调度任务。

  1. /** Ready async calls in the order they’ll be run. */

  2. private final Deque readyAsyncCalls = new ArrayDeque<>();

  3. /** Running asynchronous calls. Includes canceled calls that haven’t finished yet. */

  4. private final Deque runningAsyncCalls = new ArrayDeque<>();

《okhttp连接池复用机制》文章中我们在缓存Connection连接的时候也是使用的Deque双端队列。这里同样的方式,可以方便在队列头添加元素,移除尾部的元素。

2.2 过程分析

Call代用equeue方法的时候

  1. synchronized void enqueue(AsyncCall call) {
  2. if (runningAsyncCalls.size() < maxRequests && runningCallsForHost(call) < maxRequestsPerHost) {
  3. runningAsyncCalls.add(call);
  4. executorService().execute(call);
  5. } else {
  6. readyAsyncCalls.add(call);
  7. }
  8. }

方法中满足执行队列里面不足最大线程数maxRequests并且Call对应的host数目不超过maxRequestsPerHost 的时候直接把call对象直接推入到执行队列里,并启动线程任务(Call本质是一个Runnable)。否则,当前线程数过多,就把他推入到等待队列中。Call执行完肯定需要在runningAsyncCalls 队列中移除这个线程。那么readyAsyncCalls队列中的线程在什么时候才会被执行呢。

追溯下AsyncCall 线程的执行方法

  1. @Override
  2. protected void execute() {
  3. boolean signalledCallback = false;
  4. try {
  5. Response response = getResponseWithInterceptorChain(forWebSocket);
  6. if (canceled) {
  7. signalledCallback = true;
  8. responseCallback.onFailure(RealCall.this, new IOException(“Canceled”));
  9. } else {
  10. signalledCallback = true;
  11. responseCallback.onResponse(RealCall.this, response);
  12. }
  13. } catch (IOException e) {
  14. if (signalledCallback) {
  15. // Do not signal the callback twice!
  16. Platform.get().log(INFO, "Callback failure for " + toLoggableString(), e);
  17. } else {
  18. responseCallback.onFailure(RealCall.this, e);
  19. }
  20. } finally {
  21. client.dispatcher().finished(this);
  22. }
  23. }
  24. }

这里做了核心request的动作,并把失败和回复数据的结果通过responseCallback 回调到Dispatcher。执行操作完毕了之后不管有无异常都会进入到dispactcherfinished方法。

  1. private void finished(Deque calls, T call, boolean promoteCalls) {

  2. int runningCallsCount;

  3. Runnable idleCallback;

  4. synchronized (this) {

  5. if (!calls.remove(call)) throw new AssertionError(“Call wasn’t in-flight!”);

  6. if (promoteCalls) promoteCalls();

  7. runningCallsCount = runningCallsCount();

  8. idleCallback = this.idleCallback;

  9. }

  10. && idleCallback != null) {

  11. idleCallback.run();

  12. }

  13. }

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

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

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

img

img

img

img

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

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

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

最后看一下学习需要的所有知识点的思维导图。在刚刚那份学习笔记里包含了下面知识点所有内容!文章里已经展示了部分!如果你正愁这块不知道如何学习或者想提升学习这块知识的学习效率,那么这份学习笔记绝对是你的秘密武器!

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

112)]

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

  • 5
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值