OkHttp的任务调度

实现同步异步请求

同步:excute()
异步:enqueue()
发送的同步/异步请求都会在dispatcher中管理其状态
作用:维护请求的状态(包括同步和异步)并维护了一个线程池(更高效的执行异步请求),用于执行相应的请求。维护任务队列
通过Call(call本质就是一个Runnable)进行封装,通过dispatcher把执行的请求推到就绪请求队列中。
在这里插入图片描述

dispatcher()源码

一个线程池,两个请求队列

 private int maxRequests = 64;
  private int maxRequestsPerHost = 5;
  private Runnable idleCallback;

  /** Executes calls. Created lazily. */
  private ExecutorService executorService;

  /** Ready async calls in the order they'll be run. 
	  就绪状态的异步请求队列
  */
  private final Deque<AsyncCall> readyAsyncCalls = new ArrayDeque<>();

  /** Running asynchronous calls.正在执行的异步请求
 	  Includes canceled calls that haven't finished yet. 已经取消,但是没有执行完的异步请求
  */
  private final Deque<AsyncCall> runningAsyncCalls = new ArrayDeque<>();

  /** Running synchronous calls. Includes canceled calls that haven't finished yet.*/
  private final Deque<RealCall> runningSyncCalls = new ArrayDeque<>();

  public Dispatcher(ExecutorService executorService) {
    this.executorService = executorService;
  }

  public Dispatcher() {
  }

线程池

//线程池
  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; 
    }

0:核心线程池数量,空闲一段时间之后就会将所有的线程进行销毁
Integer.MAX_VALUE:最大的线程数,任务过来的时候可以无限的扩充线程的最大值,由于受到maxRequests限制,也不是可以无限创建的
60:当线程数大于核心线程数,多余的线程的最大存活时间
比如:开启20个并发请求,就会有20个请求,线程池就会在60s时候相继关闭所有无用的线程。

enqueue

synchronized void enqueue(AsyncCall call) {
    if (runningAsyncCalls.size() < maxRequests && runningCallsForHost(call) < maxRequestsPerHost) {
      runningAsyncCalls.add(call);
      executorService().execute(call);//线程池负责自动创建,销毁以及线程的管理
    } else {
      readyAsyncCalls.add(call);//异步请求加入就绪队列当中,进行缓存等待的操作
    }

两个队列

Dispatcher生产者,默认在主线程执行
ExecutorService,消费者池
两个队列一个用于缓存,一个用于执行任务

 /** Ready async calls in the order they'll be run. 
	  就绪状态的异步请求队列
  */
  private final Deque<AsyncCall> readyAsyncCalls = new ArrayDeque<>();

  /** Running asynchronous calls.正在执行的异步请求
 	  Includes canceled calls that haven't finished yet. 已经取消,但是没有执行完的异步请求
  */
  private final Deque<AsyncCall> runningAsyncCalls = new ArrayDeque<>();

判断最大请求数以及相同主机最大请求数判断,满足的话就可以进入到队列中立即执行,如果不满足,说明执行队列中存满了,就会进入到就绪队列中,等运行队列中有空间了,再把缓存的队列放入到执行队列当中,任务完成之后会调用promoteCall()方法,手动清除缓存区。

缓存请求的异步队列的执行时间

private <T> void finished(Deque<T> calls, T call, boolean promoteCalls) {
    int runningCallsCount;
    Runnable idleCallback;
    synchronized (this) {
      if (!calls.remove(call)) throw new AssertionError("Call wasn't in-flight!");
      if (promoteCalls) promoteCalls();
      runningCallsCount = runningCallsCount();
      idleCallback = this.idleCallback;
    }

calls.remove():把异步请求从正在执行的异步请求的队列中删除
promoteCalls():调整任务队列的,由于同步和异步都是不安全的,所以他需要在 synchronized中进行。
runningCallsCount():正在执行的异步请求和同步请求的数目总和

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java微信公众号定时发送消息模板通常分为两部分: 第一部分是实现Wechat API与Java后端的通信。Wechat API是微信提供的接口,可以让我们通过Java代码与微信公众号进行交互。具体来说,我们需要在Java中通过HTTP或HTTPS协议发送请求,获取微信公众号的access_token,再通过access_token来进行后续的操作,例如:获取用户的消息、发送消息、创建菜单等等。Java中可以使用Apache HttpClient或OkHttp等库来发送HTTP请求,拿到接口返回的JSON数据。 第二部分是实现定时发送消息。Java中可以使用Quartz框架用于实现定时任务。Quartz是一个开源的作业调度框架,可以用于在指定时间执行任务。我们可以利用Quartz提供的API,创建一个定时任务,定期调用我们的后端Java代码发送微信公众号的消息。在定时任务的实现中,我们需要考虑任务的频率、执行时间、失败策略等问题。 当以上两部分都实现以后,我们就可以在Java后端实现微信公众号定时发送消息的功能了。具体来说,我们需要将发送消息的代码放到定时任务中,指定要发送的消息内容,以及要发送给的用户。然后启动定时任务即可,Java后端会自动按照设定的规则发送微信公众号的消息。 值得注意的是,微信官方文档要求所有公众号对接的应用都必须是官方认证的,所以在实现这个功能之前,我们需要先将我们的应用向微信官方申请认证,获得相应的API调用权限。 ### 回答2: Java微信公众号定时发送消息模板是一种利用Java语言实现的微信公众平台的定时发送消息模板。该模板可以帮助微信公众号上的管理员在指定时间点自动发送指定的消息,从而提高工作效率和用户体验。 实现Java微信公众号定时发送消息模板的关键是使用了微信公众平台提供的接口。这些接口可以通过Java语言进行调用,从而实现向微信公众号发送消息、设置菜单、自动回复、素材管理、用户管理等功能。在定时发送消息模板中,我们需要利用这些接口设置定时任务,并在指定时间点调用接口发送消息。 具体实现过程如下: 1. 首先需要在微信公众平台上申请开发者账号,并获取相应的开发者ID、开发者密码和Token。 2. 在Java中使用微信公众平台提供的SDK调用相关接口,实现发送消息、设置菜单、自动回复等功能。 3. 利用Java中的定时任务框架,如Quartz、Spring定时任务等,在指定的时间点调用相应的微信接口实现定时发送消息。 4. 利用Java中的模板功能,以标准化的格式组织需要发送的消息内容,使消息具有可读性和易管理性。 需要注意的是,为了避免被微信公众平台封禁账号,我们需要遵循微信公众平台的相关规定,比如定时发送的消息不能含有敏感词汇、不能频繁发送等。另外,定时发送的频次也需要控制在一定范围内,避免对用户造成困扰。 总之,Java微信公众号定时发送消息模板是一种非常实用的工具,可以帮助管理员提高工作效率和用户体验。但实现过程需要注意一些细节和规范,确保操作的合法性和安全性。 ### 回答3: Java微信公众号定时发送消息模板是指利用Java编程语言实现微信公众号定时发送消息功能,并且提供了消息模板方便开发者进行二次开发。这个模板主要包含以下几个方面: 一、微信公众号接入 首先,需要在微信公众号后台注册并获取开发者ID和开发者密钥,然后使用Java框架接入微信公众号的开放平台。接入之后,就可以使用微信公众号提供的API实现发送消息功能。 二、定时任务 接下来,需要编写Java代码实现定时任务功能。Java提供了Timer和TimerTask类可以很方便地实现定时任务功能。开发者可以选择设定每日,周,月等不同的定时任务。 三、发送消息 定时任务设置好后,需要编写Java代码实现发送消息的功能。这个功能可以通过调用微信公众号提供的API来实现。消息的发送可以以文本消息,图文消息,视频消息等不同的形式呈现。 四、编写消息模板 最后,为了让开发者方便使用,我们需要编写一个消息模板来进行二次开发。消息模板可以包含以下几个方面:微信公众号接入的配置信息、定时任务的配置信息、发送消息的配置信息,以及发送消息的实现方法等。这个模板可以让开发者快速地实现微信公众号定时发送消息的功能。 通过这个模板,开发者可以很方便地搭建起微信公众号定时发送消息的系统,从而可以提高服务的质量和用户体验。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值