SpringBoot默认200个线程对于Websocket长连接够用吗?(一)

2023-11-11 01:08

本文主要是介绍SpringBoot默认200个线程对于Websocket长连接够用吗?(一),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

上篇推文从源码剖析SpringBoot中Tomcat的默认最大连接数中我们知道,SpringBoot的内嵌Tomcat默认的最大连接数为200。那么,这个默认值对于项目中引入了WebSocket使用长连接后,是否足够用呢?今天强哥就带大家一起从源码的角度来分析一下。

我们还是从上一篇推文给的代码入手(需要源码的小伙伴可后台回复:WebSocket获取)。强哥想了想,要判断200个线程是否够,可能并不是那么好量化,不如我们就先配置1个线程入手看看效果如何:

server.tomcat.max-threads=1

然后启动项目,前端试试来5个WebSocket请求是否可以成功连上:

没问题,都连上了。为了节省篇幅,我们直接来Chrome浏览器最大限制的WebSocket个数256:

同样也是可以,那么问题来了,为什么这一个线程就可以处理这么多个WebSocket长连接请求呢?

哈哈,想必有一些基础的小伙伴应该能想到:会不会用的是Java NIO来处理的这些请求呢?那么,我们就从源码(本篇涉及的SpringBoot源码版本为:2.2.3.BUILD-SNAPSHOT)入手先来看看这个猜想是否正确。

对于Java NIO不熟悉的同学,今天的另一篇推文强哥也搜了一篇比较优质的科普文,想了解或是复习一下的可以跳过去看看哦。

可是,源码又要从哪里入手呢?当然还是我们上面的那个配置啦。上篇文章中我们知道,按住ctrl然后鼠标点击该配置便能进入对应的ServerProperties配置类中该属性对应的set方法:

然后,找到maxThreads使用的地方:

即在customize方法中调用:

我们进到customizeMaxThreads方法中:

我们在上篇文章中也见过这个方法,重点来了,AbstractProtocol protocol这个便是我们内嵌Tomcat使用的协议。我们在上图位置打个断点然后重启项目试试:

通过Debug我们可以看出,AbstractProtocol的实现是:Http11NioProtocol,从这个类名称我们便可以很明显的看出确实使用的是NIO,从而也证明了我们前面的猜想。不过,这只是打断点看到的,没看到源码心里还是有点不痛快。怎么从源码里找到这个Http11NioProtocol被设置到内嵌Tomcat中呢。

我们还是留意这个方法,方法是调用了factory的addConnectorCustomizers方法,而该方法的作用从名称上便知道是为了设置用户自定义连接配置的。那么这个factory是什么呢,从类名ConfigurableTomcatWebServerFactory定义上,我们可以猜想,应该是生成内嵌Tomcat的工厂类。

那么,我们就进入这个类一探究竟:

哦,原来是个接口,有两个实现,用的会是哪个呢?不用多说,还是打断点看一下吧:

用的是TomcatServletWebServerFactory类,那就进去看看。哈哈,我们在类结构中可以找到一个静态属性DEFAULT_PROTOCOL,其内容恰好是我们Debug的时候protocol显示的内容:org.apache.coyote.http11.Http11NioProtocol。

而这个静态属性又被赋值给了protocol变量:

再看看这个protocol的调用:

我的天,我们看到了什么?

Tomcat tomcat = new Tomcat();

这个不就是SpringBoot内嵌生成的那个Tomcat吗?哈哈哈,原来是在这里,真是有种发现新大陆的感觉。看看他的包路径基本可以确认无疑(并不是我们上篇推文中映射自定义配置用的Tomcat类):

而那个protocol就是用在了这里啦:

Connector connector = new Connector(this.protocol);

至此,我们就能够证实只设置一个线程却可以同时连接上多个WebSocket请求的原因是由于SpringBoot的内嵌Tomcat是用了Java NIO多路复用来处理请求的。同时,我们也“幸运”地发现了SpringBoot内嵌Tomcat生成的具体位置是在TomcatServletWebServerFactory类中。

不过,爱刨根问底的小伙伴可能又会有所疑问,既然使用的是Java NIO来处理请求,那么一个线程能最多处理多少个请求呢?如果处理的不多的话,默认200个线程不是还是不够用吗?

由于篇幅问题,这些疑问就留在强哥的下一篇推文继续和大家一起探讨啦~

关注公众号获取更多内容,有问题也可在公众号提问哦:

强哥叨逼叨

叨逼叨编程、互联网的见解和新鲜事

这篇关于SpringBoot默认200个线程对于Websocket长连接够用吗?(一)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/386518

相关文章

使用Java发送邮件到QQ邮箱的完整指南

《使用Java发送邮件到QQ邮箱的完整指南》在现代软件开发中,邮件发送功能是一个常见的需求,无论是用户注册验证、密码重置,还是系统通知,邮件都是一种重要的通信方式,本文将详细介绍如何使用Java编写程... 目录引言1. 准备工作1.1 获取QQ邮箱的SMTP授权码1.2 添加JavaMail依赖2. 实现

Java嵌套for循环优化方案分享

《Java嵌套for循环优化方案分享》介绍了Java中嵌套for循环的优化方法,包括减少循环次数、合并循环、使用更高效的数据结构、并行处理、预处理和缓存、算法优化、尽量减少对象创建以及本地变量优化,通... 目录Java 嵌套 for 循环优化方案1. 减少循环次数2. 合并循环3. 使用更高效的数据结构4

java两个List的交集,并集方式

《java两个List的交集,并集方式》文章主要介绍了Java中两个List的交集和并集的处理方法,推荐使用Apache的CollectionUtils工具类,因为它简单且不会改变原有集合,同时,文章... 目录Java两个List的交集,并集方法一方法二方法三总结java两个List的交集,并集方法一

Spring AI集成DeepSeek三步搞定Java智能应用的详细过程

《SpringAI集成DeepSeek三步搞定Java智能应用的详细过程》本文介绍了如何使用SpringAI集成DeepSeek,一个国内顶尖的多模态大模型,SpringAI提供了一套统一的接口,简... 目录DeepSeek 介绍Spring AI 是什么?Spring AI 的主要功能包括1、环境准备2

Spring AI集成DeepSeek实现流式输出的操作方法

《SpringAI集成DeepSeek实现流式输出的操作方法》本文介绍了如何在SpringBoot中使用Sse(Server-SentEvents)技术实现流式输出,后端使用SpringMVC中的S... 目录一、后端代码二、前端代码三、运行项目小天有话说题外话参考资料前面一篇文章我们实现了《Spring

Spring AI与DeepSeek实战一之快速打造智能对话应用

《SpringAI与DeepSeek实战一之快速打造智能对话应用》本文详细介绍了如何通过SpringAI框架集成DeepSeek大模型,实现普通对话和流式对话功能,步骤包括申请API-KEY、项目搭... 目录一、概述二、申请DeepSeek的API-KEY三、项目搭建3.1. 开发环境要求3.2. mav

Springboot的自动配置是什么及注意事项

《Springboot的自动配置是什么及注意事项》SpringBoot的自动配置(Auto-configuration)是指框架根据项目的依赖和应用程序的环境自动配置Spring应用上下文中的Bean... 目录核心概念:自动配置的关键特点:自动配置工作原理:示例:需要注意的点1.默认配置可能不适合所有场景

使用Apache POI在Java中实现Excel单元格的合并

《使用ApachePOI在Java中实现Excel单元格的合并》在日常工作中,Excel是一个不可或缺的工具,尤其是在处理大量数据时,本文将介绍如何使用ApachePOI库在Java中实现Excel... 目录工具类介绍工具类代码调用示例依赖配置总结在日常工作中,Excel 是一个不可或缺的工http://

Java8需要知道的4个函数式接口简单教程

《Java8需要知道的4个函数式接口简单教程》:本文主要介绍Java8中引入的函数式接口,包括Consumer、Supplier、Predicate和Function,以及它们的用法和特点,文中... 目录什么是函数是接口?Consumer接口定义核心特点注意事项常见用法1.基本用法2.结合andThen链

spring @EventListener 事件与监听的示例详解

《spring@EventListener事件与监听的示例详解》本文介绍了自定义Spring事件和监听器的方法,包括如何发布事件、监听事件以及如何处理异步事件,通过示例代码和日志,展示了事件的顺序... 目录1、自定义Application Event2、自定义监听3、测试4、源代码5、其他5.1 顺序执行