Tomcat 7.0.32 +Spring MVC Servlet 3 异步不起作用

2024-01-09

我编写了非常简单的控制器来测试 Servlet 3 功能:

@Autowired
    ThreadPoolTaskExecutor taskExecutor;

    @RequestMapping(value="{name}", method = RequestMethod.GET)
    public @ResponseBody DeferredResult<MyResponse> getShopInJSON(@PathVariable String name) {

        DeferredResult<MyResponse> df = new DeferredResult<MyResponse>();
        taskExecutor.submit(new MyRunnable(df));    

        return df; 
    }

在单独的线程中,我只执行 5 秒睡眠命令,然后返回MyResultPOJO 至DeferredResult.

我的 web.xml 文件符合 Servlet 3 规范:

<web-app xmlns="http://java.sun.com/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
                             http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
         version="3.0"
         metadata-complete="true">
  <display-name>Archetype Created Web Application</display-name>
    <servlet>
    <async-supported>true</async-supported>
    <servlet-name>mvc-dispatcher</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <load-on-startup>1</load-on-startup>
  </servlet>
  <servlet-mapping>
    <servlet-name>mvc-dispatcher</servlet-name>
    <url-pattern>/</url-pattern>
  </servlet-mapping>
</web-app>

我的连接器 tomcat 如下:

 <Connector port="8080" protocol="HTTP/1.1"
                maxThreads="5"
                acceptCount="5"
                connectionTimeout="20000"
                redirectPort="8443" />

现在这是有趣的部分。当运行打开 10 个并发连接的简单程序时,我发现第一个连接只提供了 5 个连接,而在第一个连接被释放后,第二个 5 个连接才被提供服务(您可以从时间茎中看到它)。这不是 Servlet 3.0 应该有的行为

Fri May 31 01:17:57 IDT 2013: Preparing 10 concurrent connections
Fri May 31 01:18:02 IDT 2013: Output from Server int thread 9 :{"props1":"param1","props2":"param1"}
Fri May 31 01:18:02 IDT 2013: Output from Server int thread 8 :{"props1":"param1","props2":"param1"}
Fri May 31 01:18:02 IDT 2013: Output from Server int thread 4 :{"props1":"param1","props2":"param1"}
Fri May 31 01:18:02 IDT 2013: Output from Server int thread 7 :{"props1":"param1","props2":"param1"}
Fri May 31 01:18:02 IDT 2013: Output from Server int thread 2 :{"props1":"param1","props2":"param1"}
Fri May 31 01:18:07 IDT 2013: Output from Server int thread 1 :{"props1":"param1","props2":"param1"}
Fri May 31 01:18:07 IDT 2013: Output from Server int thread 0 :{"props1":"param1","props2":"param1"}
Fri May 31 01:18:07 IDT 2013: Output from Server int thread 5 :{"props1":"param1","props2":"param1"}
Fri May 31 01:18:07 IDT 2013: Output from Server int thread 6 :{"props1":"param1","props2":"param1"}
Fri May 31 01:18:07 IDT 2013: Output from Server int thread 3 :{"props1":"param1","props2":"param1"}

如果将 Tomcat Connector 更改为

   <Connector connectionTimeout="200000" maxThreads="5" port="8080" protocol="org.apache.coyote.http11.Http11NioProtocol" redirectPort="8443"/>

它就像魅力一样起作用。我不想这样做。根据 Tomcat 文档,我应该接收 Servlet 3.0 功能,而无需Http11NioProtocol连接器。

怎么了?


问题是由于最大线程数=5在 Tomcat 配置中进行设置。

对于非NIO的情况,这个设置不仅限制了最大请求处理线程数,还限制了最大连接数!

由于您尚未指定 maxConnections,因此它为 maxConnections 选择默认值。 这是摘录自汤姆猫文档 http://tomcat.apache.org/tomcat-7.0-doc/config/http.html关于它如何选择 maxConnections 的默认值:

maxConnections :服务器的最大连接数 将在任何给定时间接受并处理。当这个数字被 达到后,服务器将不再接受任何连接,直到 连接数低于该值。操作系统可能 仍然接受基于acceptCount 设置的连接。默认 值因连接器类型而异。对于 BIO,默认值是 最大线程数除非使用执行器,在这种情况下默认将 是来自执行器的 maxThreads 值。对于 NIO,默认值为 10000。对于 APR/native,默认值为 8192。

您可以显式指定一个最大连接数=“10”(例如)设置覆盖此默认行为。然后,您应该看到无论使用什么连接器,都可以处理 10 个并行请求。我尝试过这个并且有效。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Tomcat 7.0.32 +Spring MVC Servlet 3 异步不起作用 的相关文章

随机推荐