关于tomcat的apr、bio、nio模式

关于tomcat的这几种模式,以及在springboot内嵌tomcat的方式下如何设置,可以参考以下几篇文章进行设置。

首先将springboot应用程序跑起来,用postman随便访问一个接口可以看到打印出来的记录里显示出是基于nio模式的

springboot 使用 本地apollo配置 springboot apr_tomcat

接下来我们参考上面的这些文章,将springboot程序设置为Apr模式运行。关键的几个步骤见下面的

注意启用apr模式之前,相关的apr要安装。具体安装方法很简单,见这篇文章 [Tomcat中如何配置使用APR],为了方便懒人,直接截图过来了,()。感谢原作者。

springboot 使用 本地apollo配置 springboot apr_tomcat_02


附图片中的apr下载路径apr下载路径 设置tomcat模式为 apr 模式。这一步都是so easy

springboot 使用 本地apollo配置 springboot apr_tomcat_03

好了,一切都已经搞定了。应用程序跑起来就是这个样子的了

springboot 使用 本地apollo配置 springboot apr_apache_04


已经根据Apr模式启动了。一切看起来都很OK。走到这一步,基本上都没啥问题了。

接下来让人郁闷的事情来了

问题现象

用POSTMAN 访问127.0.0.1 没有任何响应!见下图

springboot 使用 本地apollo配置 springboot apr_apache_05

因为只改动了这个protocol设置。那么注销掉下图红色这一行的设置(这行代码是设置protocol模式为 spring.server.protocol=org.apache.coyote.http11.Http11AprProtocol 的)。

springboot 使用 本地apollo配置 springboot apr_spring_06


注销掉红色这部分的代码,发现应用以nio模式启动,正常接收和响应数据。

说明问题的确出在apr模式上。但是为什么apr模式会有问题,nio模式没有问题呢?

问题描述

1.nio模式没有问题,可以正常响应和输出数据。只要切换到apr模式就会不响应。内心是这样的

springboot 使用 本地apollo配置 springboot apr_spring_07

分析思路
1.请求数据发过去了没有,tomcat有没有接收到
2.端口有没有绑定(做过socket编程的小伙伴们就有了解了,socket需要create、bind、accept、listen之类的操作)有没有正常监听?

因为我这个springboot应用程序指定的端口是8102。用cmd netstat命令查看了一下绑定的端口

这个是在apr模式下:

springboot 使用 本地apollo配置 springboot apr_spring_08


可以看到8102端口有在正常的listening状态。但是ip地址有些诡异。

再切换回nio模式

springboot 使用 本地apollo配置 springboot apr_tomcat_09

看出区别了没有?
在NIO模式下,会有两个地址绑定,一个是IP4,一个是IP6。而在APR模式下只有一个IP6的地址绑定。那么问题就在于tomcat的地址绑定上。为什么NIO模式会有IP4和IP6,而APR模式只有IP6呢?导致用本机地址127.0.0.1无法访问。

于是穷追不舍的我就跟到了springboot内嵌的tomcat源码里面去了。手动滑稽。

nio模式 bind相关的代码

这个是NIO模式的bind函数,在这个路径

C:\Users\Administrator.gradle\caches\modules-2\files-2.1\org.apache.tomcat.embed\tomcat-embed-core\8.5.31\5ade5e787bcaabb9d088d8e4b22a8edffa43a783\tomcat-embed-core-8.5.31-sources.jar!\org\apache\tomcat\util\net\NioEndpoint.java

绑定的 0.0.0.0 地址

springboot 使用 本地apollo配置 springboot apr_tomcat_10

它的地址是怎么来的呢??

仔细看下它获取ip地址的业务逻辑是一个三目运算

InetSocketAddress addr = 
(getAddress()!=null?new InetSocketAddress(getAddress(),getPort()):new InetSocketAddress(getPort()));

如果没有指定地址,呢么就

new InetSocketAddress(getPort()))

跟进去看
C:\Program Files\Java\jdk1.8.0_151\src.zip!\java\net\InetSocketAddress.java

springboot 使用 本地apollo配置 springboot apr_tomcat_11

主要调用了一个anyLocalAddress 字面意思理解就是任意的物理地址

springboot 使用 本地apollo配置 springboot apr_tomcat_12


anyLocalQAddress地址就是 0.0.0.0

springboot 使用 本地apollo配置 springboot apr_tomcat_13


所以,综上所述,在NIO模式下,不需要指定地址,它会自动默认一个 0.0.0.0的地址

apr模式 bind相关的代码

这个是APR模式的bind函数,在这个路径

C:\Users\Administrator.gradle\caches\modules-2\files-2.1\org.apache.tomcat.embed\tomcat-embed-core\8.5.31\5ade5e787bcaabb9d088d8e4b22a8edffa43a783\tomcat-embed-core-8.5.31-sources.jar!\org\apache\tomcat\util\net\AprEndpoint.java

springboot 使用 本地apollo配置 springboot apr_apache_14


我们可以看到,它调用 getAddress() ,因为没有手动设置address地址,所以返回值是空。接下来的addressStr也是空的。然后在后面的绑定中也是基于这个参数去绑定。自然是没有地址了。

springboot 使用 本地apollo配置 springboot apr_tomcat_15

断点走过了listen之后,cmd命令去看,就已经绑定了端口。但是没有监听地址

springboot 使用 本地apollo配置 springboot apr_spring_16

结果分析

1.在NIO模式下没有指定地址的情况,会调用

new InetSocketAddress(getPort())) 去获取一个 anyLocalAddress (这个参数启动时候会被初始化为 0.0.0.0),所以NIO模式下不用指定地址即可绑定。

2.在apr模式下没有指定地址的情况,会继续往前执行,看这个

Address.info函数,传入的addressStr参数是空的。

springboot 使用 本地apollo配置 springboot apr_spring_17


Address.info是一个native函数,也就是jni调用。对接到系统层了,无法源码调试。但是看函数的注释,也很清楚,如果没有地址没有指定,它会给你返回一个 0.0.0.0 或 :: 也就是我们看到的绑定的地址是 ::

springboot 使用 本地apollo配置 springboot apr_tomcat_18

真相大白了。没有指定地址的情况下,的确是会bind到 :: 这个地址。

解决方案

在apr配置中手工指定地址。

springboot 使用 本地apollo配置 springboot apr_tomcat_19


springboot 使用 本地apollo配置 springboot apr_tomcat_20

再次启动,tomcat绑定地址

springboot 使用 本地apollo配置 springboot apr_apache_21

apr模式启动且接口产生响应。

总结

解决问题后,我没有非常深入去探究为什么没有指定address的情况下,它绑定的地址会返回 :: (是不是哪个参数没有设置?跟我本机安装了vmware产生多个虚拟网卡有没有关系?跟打开ipv6有没有关系?跟内置tomcat默认设置有没有关系等)