前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >okhttp之Dispatcher

okhttp之Dispatcher

作者头像
103style
发布2022-12-19 13:43:15
3210
发布2022-12-19 13:43:15
举报

转载请以链接形式标明出处: 本文出自:103style的博客


base on 3.12.0


目录

  • 简介
  • Dispatcher成员变量介绍
  • Dispatcher构造方法介绍
  • Dispatcher主要方法介绍
  • 小结

简介

首先我们来介绍下 Dispatcher,官方描述是这样的:

Policy on when async requests are executed. 执行异步请求时的策略

所以Dispatcher是我们进行异步请求是 okhttp 给我们提供的 执行异步请求时的策略.

代码语言:javascript
复制
public final class Dispatcher {...}

我们可以看到 Dispatcher 类是由 final 修饰的,代表它不能被继承。

我们先来看下 Dispatcher是什么时候设置的。 通过查看下面 OkHttpClient 的代码,我们知道在我们创建 OkHttpClient 的时候,如果我们没有通过 builder.dispatcher(Dispatcher dispatcher) 修改 Dispatcher的配置的话,默认的 dispatcher 就是 默认配置的 Dispatcher类。

代码语言:javascript
复制
public class OkHttpClient implements ... {
    ...
    public Builder newBuilder() {
        return new Builder(this);
    }
    public static final class Builder {
        Dispatcher dispatcher;
        ...
        public Builder() {
            dispatcher = new Dispatcher();
            ...
        }
        public Builder dispatcher(Dispatcher dispatcher) {
            if (dispatcher == null) throw new IllegalArgumentException("dispatcher == null");
            this.dispatcher = dispatcher;
            return this;
        }
    }
}

Dispatcher成员变量介绍

  • int maxRequests = 64; 默认同时执行的最大请求数, 可以通过setMaxRequests(int)修改.
  • int maxRequestsPerHost = 5; 每个主机默认请求的最大数目, 可以通过setMaxRequestsPerHost(int)修改.
  • private @Nullable Runnable idleCallback; 调度没有请求任务时的回调.
  • ExecutorService executorService; 执行异步请求的线程池,默认是 核心线程为0,最大线程数为Integer.MAX_VALUE,空闲等待为60s.
  • Deque<AsyncCall> readyAsyncCalls = new ArrayDeque<>(); 异步请求的执行顺序的队列.
  • Deque<AsyncCall> runningAsyncCalls = new ArrayDeque<>(); 运行中的异步请求队列.
  • Deque<RealCall> runningSyncCalls = new ArrayDeque<>(); 运行中的同步请求队列.

Dispatcher构造函数

代码语言:javascript
复制
public Dispatcher(ExecutorService executorService) {
  this.executorService = executorService;
}
public Dispatcher() {}

可以自己设置执行任务的线程池。


Dispatcher主要方法介绍

配置和获取 同时执行请求的最大任务数、同主机允许同时执行的最大任务数

代码语言:javascript
复制
public void setMaxRequests(int maxRequests) {...}
public synchronized int getMaxRequests() {...}
public void setMaxRequestsPerHost(int maxRequestsPerHost) {...}
public synchronized int getMaxRequestsPerHost() {...}

设置没有请求任务时的回调

代码语言:javascript
复制
public synchronized void setIdleCallback(@Nullable Runnable idleCallback) {
    this.idleCallback = idleCallback;
}

添加到请求队列

代码语言:javascript
复制
void enqueue(AsyncCall call) {
    synchronized (this) {
        readyAsyncCalls.add(call);
    }
    promoteAndExecute();
}
synchronized void executed(RealCall call) {
    runningSyncCalls.add(call);
}

执行等待队列中的请求任务

代码语言:javascript
复制
private boolean promoteAndExecute() {
    assert (!Thread.holdsLock(this));
    List<AsyncCall> executableCalls = new ArrayList<>();
    boolean isRunning;
    synchronized (this) {
        //获取等待中的任务队列
        for (Iterator<AsyncCall> i = readyAsyncCalls.iterator(); i.hasNext(); ) {
            AsyncCall asyncCall = i.next();
            // 超过可以同时运行的最大请求任务数
            if (runningAsyncCalls.size() >= maxRequests) break; 
            // 超过同一主机同时运行的最大请求任务数
            if (runningCallsForHost(asyncCall) >= maxRequestsPerHost) continue; 
            i.remove();
            executableCalls.add(asyncCall);
            runningAsyncCalls.add(asyncCall);
        }
        isRunning = runningCallsCount() > 0;
    }
    for (int i = 0, size = executableCalls.size(); i < size; i++) {
        AsyncCall asyncCall = executableCalls.get(i);
        asyncCall.executeOn(executorService());
    }
    return isRunning;
}

获取执行任务的线程池. 如果没有通过构造方法Dispatcher(ExecutorService executorService) 设置线程池的话,默认就是 核心线程为0,最大线程数为Integer.MAX_VALUE,空闲等待为60s,用SynchronousQueue保存等待任务 的线程池。

代码语言:javascript
复制
public synchronized ExecutorService executorService() {
    if (executorService == null) {
        executorService = new ThreadPoolExecutor(0, Integer.MAX_VALUE, 60, TimeUnit.SECONDS,
                new SynchronousQueue<Runnable>(), Util.threadFactory("OkHttp Dispatcher", false));
    }
    return executorService;
}

获取和当前请求的host一样的运行中的请求个数.

代码语言:javascript
复制
private int runningCallsForHost(AsyncCall call) {
    int result = 0;
    for (AsyncCall c : runningAsyncCalls) {
        if (c.get().forWebSocket) continue;
        if (c.host().equals(call.host())) result++;
    }
    return result;
}

取消所有请求任务

代码语言:javascript
复制
public synchronized void cancelAll() {
    for (AsyncCall call : readyAsyncCalls) {
        call.get().cancel();
    }
    for (AsyncCall call : runningAsyncCalls) {
        call.get().cancel();
    }
    for (RealCall call : runningSyncCalls) {
        call.cancel();
    }
}

结束请求任务

代码语言:javascript
复制
void finished(AsyncCall call) {}
void finished(RealCall call) {}
private <T> void finished(Deque<T> calls, T call) {
    Runnable idleCallback;
    synchronized (this) {
        if (!calls.remove(call)) throw new AssertionError("Call wasn't in-flight!");
        idleCallback = this.idleCallback;
    }
    boolean isRunning = promoteAndExecute();

    if (!isRunning && idleCallback != null) {
        idleCallback.run();
    }
}

获取等待中、执行中的请求任务

代码语言:javascript
复制
public synchronized List<Call> queuedCalls() {
    List<Call> result = new ArrayList<>();
    for (AsyncCall asyncCall : readyAsyncCalls) {
        result.add(asyncCall.get());
    }
    return Collections.unmodifiableList(result);
}
public synchronized List<Call> runningCalls() {
    List<Call> result = new ArrayList<>();
    result.addAll(runningSyncCalls);
    for (AsyncCall asyncCall : runningAsyncCalls) {
        result.add(asyncCall.get());
    }
    return Collections.unmodifiableList(result);
}
public synchronized int queuedCallsCount() {
    return readyAsyncCalls.size();
}
public synchronized int runningCallsCount() {
    return runningAsyncCalls.size() + runningSyncCalls.size();
}

小结

Dispatcher是我们进行异步请求是 okhttp 给我们提供的 执行异步请求时的策略

Dispatcher因为是final修饰的类,所以我们我不能继承它,但是我们可以通过创建一个Dispatcher对象,然后修改 执行任务的线程池最大并发数同主机最大并发数 等。

执行任务的方法是promoteAndExecute().


以上

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2019-09-26,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 目录
  • 简介
  • Dispatcher成员变量介绍
  • Dispatcher构造函数
  • Dispatcher主要方法介绍
  • 小结
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档