从源码剖析SpringBoot中Tomcat的默认最大连接数

上篇推文为什么你的websocket只能建立256个连接?推出后,有许多小伙伴问:关键是怎么解决256这个问题。嗯,可能是我的标题起的有点问题,不过如果有认真阅读文章的话,应该会知道,其实256的限制是Chrome浏览器对WebSocket进行的限制。不过,对于一个客户端来说,假如一个窗口建立一个WebSocket连接的话,想要达到256个连接,需要打开256个窗口,这个一般来说很少出现。所以大家并不需要太多的担心这个前端限制。

不过,谈到请求连接数限制的问题,我们倒是可以将注意力转移到后端的配置上来。上篇推文中有提到在后端配置文件有如下配置用于定义tomcat请求连接数的:

server.tomcat.max-connections=10000

我们设置为10000可见已经挺大的了。但是,当请求由前端到达后端时,除了连接数与之有关外,还有一个关键的配置那就是线程数。tomact设置允许的最大线程数,也在很大程度上影响着我们后端能够处理的请求个数。

不知道大家在开发过程中有没有这种感觉,当我们真的用上新的框架或功能的时候,遇到一些属性参数,虽然我们有进行配置,但是,还是会希望了解在框架中对应配置项的默认值是多少。尤其是SpringBoot这类“约定优于配置”原则的框架,更需要我们花时间去学习。

强哥个人觉得能够保有这种追根溯源的精神,对我们技术的提升还是很有帮助的。毕竟只会CV的码农太多了,而对于CV回来的东西,无法做到胸有成竹的话,当遇到问题时,只会让我们手足无措。

所以,既然我们考虑到了性能问题,那么当我们没有配置最大请求连接数和最大线程数时,SpringBoot默认对这两个属性的配置值又是多少呢?今天强哥就带着大家一起从源码的角度来探寻一番。不过由于SpringBoot发展较为迅速,我们先从稍老的版本说起,毕竟很多公司的项目用SpringBoot2以下的还比比皆是。强哥接下来先带大家查看SpringBoot1.5.9.RELEASE版本的源码。之后再会提到新版本的改动。

那么从哪里入手进行源码追溯呢?于其像无头苍蝇一样到百度上胡乱搜索不如我们从现有的内容出发。那现在有什么呢?没错就是server.tomcat.max-connections=10000这个配置了。使用IDEA的小伙伴,直接按住Ctrl键再移动鼠标到该配置,鼠标左键点击后,便会跳转到如下类方法中:

ServerProperties是SpringBoot为我们提供的一个配置类,这里也简单提一下SpringBoot中的属性配置类。即利用@ConfigurationProperties注解,获取到配置文件中的内容,然后映射到我们的类中:

从上面的代码我们便可以得知,ServerProperties是用于映射配置文件中以server开头的属性。同理可知,既然server找到映射了,那么server.tomcat.max-connections=10000中的tomcat呢?很简单,就是作为成员变量配置在ServerProperties类中啦:

你没看错,哈哈,居然还有Jetty相关的内容,是不是发现了新大陆一般。

有的小伙伴在看到这段代码时肯定会有所联想:Tomcat tomcat = new Tomcat(),嗯?这个Tomcat难不成就是SpringBoot自带的那个内嵌Tomcat?原来它的实现就在这里??额,秀儿,请你先坐下。

其实,这个Tomcat并不是什么所谓的内嵌Tomcat,而只是ServerProperties的内部类而已。那么对他设置值又有什么用呢?这就是SpringBoot的精妙之处,通过我们配置文件中配置的server.tomcat.max-connections=10000便能够拿用户自定义的属性值去替换框架默认的属性值。

好的,了解了这点之后,顺藤摸瓜,我们便可以猜到,max-connections肯定就是Tomcat这个内部类的内部属性了:

哈哈,果然没错,可是默认值居然是0!!什么情况?难不成SpringBoot默认最大连接数为0,这不可能啊,那不是什么请求都收不到了。既然默认设置了0,肯定有其他地方对其进行了修改。

带着疑问找一下,在Tomcat类中,除了对该属性的get,set方法(我们配置文件中配置的属性即调用的这个方法对该值就行修改)之外,我们可以在Tomcat类中的customizeTomcat()方法中,找到对maxConnection的修改:

点进该方法看看:

关键在这句代码:

protocol.setMaxConnections(Tomcat.this.maxConnections);

可见,它就是将我们在配置文件中自定义配置的max-connections的值在这里进行真正的赋值使用的。那么可想而知,默认值就是在protocol之中。我们继续往上摸:

哦吼,setMaxConnections中对endpoint又进行了赋值,那下一个目标就是endpoint咯,endpoint是个成员属性:

 

那就继续进入AbstractEndpoint,找与maxConnections相关内容:

哈哈哈,终于我们摸到瓜啦,从上面代码可以看出,maxConnections的默认值就是10000。不容易啊,历经九九八十一难终于修得正果。为自己呱唧呱唧!!!

同理,上下再找找,发现maxThreads也是在此类中设置了默认值:

默认值是200哦。

到此,我们便完成了对SpringBoot 1.5.9.RELEASE版本中tomcat默认最大连接数的源码追寻过程。那么,为什么要特地提到版本号呢?当然是因为新版的会有所不同了。强哥也在SpringBoot 2.2.3.BUILD-SNAPSHOT中找了一下,这两个配置值直接就在ServerPropertise中进行定义了:

同样默认最大线程数还是200。

到了这里,可能大家又会有疑问了,maxConnections最大值10000可以理解,毕竟一个服务同时保持10000个连接确实资源也就消耗的差不多了,可是,maxThreads却只有200,是不是少了点。加入了WebSocket之后,都是长连接,那不就更不够了?

这个问题就留给下一篇推文吧,今天就先到这里了

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

强哥叨逼叨

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

发布了55 篇原创文章 · 获赞 72 · 访问量 26万+

猜你喜欢

转载自blog.csdn.net/seanxwq/article/details/104055743
今日推荐