网友直呼:DeferredResult是Spring对Servlet异步处理的包装吗?- 第416篇

关历史文章(阅读本文前,您可能需要先看下之前的系列👇

国内最全的Spring Boot系列之四

享元模式:共享女友 - 第355篇

SpringBoot 使用validation数据校验之国际化问题怎么搞?满满的干货,值得收藏 - 第411篇

什么是轮询、长轮询、长连接一篇文章让你不在懵懂 - 第412篇

Spring Boot使用Servlet居然也可以实现长轮询,敲了5年代码,我居然不知道 - 第413篇Spring Boot使用Spring DeferredResult实现长轮询纵享新丝滑让你体验丝滑般的感觉 - 第414篇

Spring Boot使用Callable和WebAsyncTask实现长轮询,战斗力杠杠的,这一节知识点满满的 - 第415篇

悟纤:师傅,公众号有人留言了。

师傅:具体说了什么?

悟纤:对方问“我记得这个DeferredResult 是 spring 对 http 异步处理的包装吧“

师傅:这个为师也不知道,徒儿,这次交给你去探索下吧。

悟纤:师傅,你觉得徒儿能搞定吗?

师傅:在没有搞之前,不要怀疑自己不能搞定。

悟纤:师傅说的是,那徒儿去探索探索。

导读

       在公众号有粉丝留言:我记得这个 DeferredResult 是 spring 对 http 异步处理的包装吧❓❓❓

       首先感谢该粉丝对博主的关注,看了下关注时间2017-06-08,是老粉了,必须得认真的关注好好解答一番,特意撰文以此表示感谢「恍如隔世」。

       本文将通过源码层面进行探索DeferredResult到底是否是Spring对servlet异步处理的包装?

              长轮询系列:

(1)✅《什么是轮询、长轮询、长连接一篇文章让你不在懵懂

(2)✅《Spring Boot使用Servlet居然也可以实现长轮询

(3)✅《Spring Boot使用Spring DeferredResult实现长轮询,纵享新丝滑让你体验丝滑般的感觉

(4)✅《Spring Boot使用Spring Callable和WebAsyncTask实现长轮询,战斗力杠杠的

(5)✅《Spring Boot使用Future+@Async实现长轮询

(6)✅《网友直呼:DeferredResult是Spring对Servlet异步处理的包装吗?》​

这一节我们来看看《网友直呼:DeferredResult是Spring对Servlet异步处理的包装吗?》

一、问题的还原

       这个问题是来自前面一篇文章《Spring Boot使用Spring DeferredResult实现长轮询纵享新丝滑让你体验丝滑般的感觉 - 第414篇》的留言,没看过的可以关注公众号「SpringBoot」,回复关键词「DeferredResult」或者「414」进行查看。

       我们在前面说道了对于Spring DeferredResult的使用很简单:

(1)声明一个变量DeferredResult,然后方法返回DeferredResult:

(2)客户端请求映射到控制器方法返回值为DeferredResult时,会立即释放Tomcat线程并将请求挂起,直到调用setResult()方法或者超时,才会响应客户端请求。

       那么底层到底做了什么呢?到底是不是对Servlet异步处理的包装呢?今天悟纤将带你一探究竟。

二、悟纤带你一探究竟

2.1 一切的缘起DispatcherServlet

悟纤(Thinking):在Controller的这个方法中,很简单,好像没啥特殊的地方,那么这个核心的代码到哪里去了呐?

悟纤(灵光一闪):在类上不是使用了@RestController,这个是@Controller和@ResponseBody的组合注解,方法上又使用了@GetMapping注解,这不是标准的Spring MVC编码方式吗?

悟纤(回想):想啊想… 我记得师傅师傅和说过,Spring MVC有一个前端控制器来着,叫DispatcherServlet,所有的请求都会经过这个进行处理。

(这一块的知识要是不知道的话,那么要补补Spring MVC的知识了)

2.2 DispatcherServlet.doDispatch()

       首先请求会进入到doService然后doDispatcher,这些都是spring mvc的知识,篇幅有限不展开细讲,看下doDispatcher:

说明:在这里获取到了处理器和处理器适配器。

       接着往下看可以看到处理器的处理的代码是:

       从List<HandlerAdapter>handlerAdapters 进行超找合适的适配器

{RequestMappingHandlerAdapter}、{HandlerFunctionAdapter} 、{HttpRequestHandlerAdapter}、{SimpleControllerHandlerAdapter}

这里找到的是RequestMappingHandlerAdapter,所以处理ha.handle()也就是

RequestMappingHandlerAdapter.handle()方法,进入到这个方法去看下:

2.3 RequestMappingHandlerAdapter.handle()

       进入代码我们发现handler()是RequestMappingHandlerAdapter的父类AbstractHandlerMethodAdapter来实现了:

       父类好像没有具体的实现而是交给了抽象方法handleInternal(),也就是子类RequestMappingHandlerAdapter来实现了:
 

       这里最重要的方法是invokeHandlerMethod(request, response,handlerMethod),通过名字就是调用handler的方法,handler在这里不就是Controller这个类吗。

看下invokeHandlerMethod组装了一堆信息,然后调用了invocableMethod.invokeAndHandle(webRequest,mavContainer),这个invocableMethod是类ServletInvocableHandlerMethod的对象,进入到这个方法看下:

       然后又进入到了handleReturnValue:

selectHandler(returnValue, returnType)

这个方法很重要,就是通过返回类型获取到的处理类

DeferredResult的处理类是DeferredResultMethodReturnValueHandler

       这里找到的类是DeferredResultMethodReturnValueHandler:

       到这里我们来梳理下核心的逻辑就是通过方法返回值的类型为DeferredResult来找到对应的Handler为DeferredResultMethodReturnValueHandler。

       如果能找到这一部分的,那么下面的就简单了。

2.4 DeferredResultMethodReturnValueHandler

       找到了对应的处理器就进入到了这个利器的handleReturnValue:

       WebAsyncManager.startDeferredResultProcessing()核心的代码:

       这里的核心就是:

(1)startAsyncProcessing(processingContext):开启异步处理。

(2)deferredResult.setResultHandler:设置deferredResult.setResult的回调。

2.5 startAsyncProcessing

       startAsyncProcessing()这个方法最终是进入到了StandardServletAsyncWebRequest的startAsync():

       熟悉的面孔,熟悉的代码。

       自此网友的疑问就解答了。

2.6 调用setResult

       当我们的异步任务执行完成后,会调用DeferredResult的setResult()方法:

设置了DeferredResultHandler,

因此会调用setConcurrentResultAndDispatch(WebAsyncManager):

       this.asyncWebRequest.dispatch() 请求调度,就是模拟客户端再次向服务器端发起请求:

       重新发起请求的话,就会回到了DispatcherServlet,到哪里进行处理呢?在RequestMappingHandlerAdapter的invokeHandlerMethod,有这么一段代码:

       这里的我们要看下wrapConcurrentResult()这个方法:

       构建了ConcurrentResultHandlerMethod,是ServletInvocableHandlerMethod的内部类,这里的逻辑创建了Callable,调用Callable的call()方法,得到返回结果,再使用返回值处理器处理返回值。

悟纤小结

(1)控制器中定义方法返回值为DeferredResult,由DeferredResultMethodReturnValueHandler处理返回结果,开启异步处理并设置DeferredResultHandler。

(2)务执行完成后调用setResult()方法,紧接着回调DeferredResultHandler的handleResult()设置结果并调度请求

(3)创建Callable对象并设置调用方法为call(),通过反射方式调用call()得到返回值,使用返回值处理器处理返回值。

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
使用Python绘制混淆矩阵,可以使用sklearn.metrics包中的confusion_matrix函数。首先,需要将预测结果和真实标签以类似的格式赋值给y_pred和y_true变量。然后,可以使用confusion_matrix函数生成混淆矩阵C,可以通过labels参数指定类别的标签。接下来,可以使用matplotlib.pyplot中的函数绘制矩阵图,使用plt.matshow(C, cmap=plt.cm.Reds)来展示混淆矩阵的颜色。可以使用plt.annotate函数在矩阵图中显示每个元素的值。最后,可以使用plt.xlabel和plt.ylabel函数设置x轴和y轴的标签。最后,使用plt.show函数显示绘制好的混淆矩阵图。 [1 [2] 示例代码如下: ```python from sklearn.metrics import confusion_matrix import matplotlib.pyplot as plt y_pred = [] # 预测结果 y_true = [] # 真实标签 C = confusion_matrix(y_true, y_pred, labels=['0','1','2','3','4']) plt.matshow(C, cmap=plt.cm.Reds) for i in range(len(C)): for j in range(len(C)): plt.annotate(C[j, i], xy=(i, j), horizontalalignment='center', verticalalignment='center') plt.ylabel('True label') plt.xlabel('Predicted label') plt.show() ``` 这段代码会根据给定的预测结果和真实标签生成混淆矩阵,并使用矩阵图展示混淆矩阵的颜色。每个元素表示预测为某个类别的样本数量。通过调整代码中的参数和标签,可以根据不同的需求进行自定义。 [1 [2 [3<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [详解使用python绘制混淆矩阵confusion_matrix)](https://download.csdn.net/download/weixin_38580959/12861679)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] - *2* *3* [利用python绘制混淆矩阵](https://blog.csdn.net/weixin_43818631/article/details/121309660)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

悟纤

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值