【发布时间】:2013-05-26 16:02:49
【问题描述】:
我写了一个非常简单的控制器来测试 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 秒的睡眠命令,之后我将 MyResult POJO 返回到 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 个连接(您可以从 time stemps 看到它)。这不是 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 connectionTimeout="200000" maxThreads="5" port="8080" protocol="org.apache.coyote.http11.Http11NioProtocol" redirectPort="8443"/>
它就像魅力一样。我不想这样做。根据 Tomcat 文档,我应该收到没有 Http11NioProtocol 连接器的 Servlet 3.0 功能。
怎么了?
【问题讨论】:
-
tikalk.com/java/servlet-3-support-spring-mvc-32 谈到了你提到的同样的问题。
-
@brainOverflow,谢谢,但我是这篇文章的作者 :) 我只是认为这种行为是不正常的,根据 Tomcat 文档 Servlet 3 也应该适用于 BIO。跨度>
-
哦,好吧...我只是想通过引用该帖子来表示对您的主张的一些支持... :-) 顺便说一句,我遇到了另一个post,他们说:@ 987654331@您是否尝试过使用 AsyncContext 而不是 DeferredResult ?可能是春天在这里引入了一些行为!
-
@brainOverflow,DeferredResult 只是包装器。它具有 AsyncContext 并在请求上调用 startAsync() 方法。使用 NIO 连接器行为正确的事实告诉我们,它更有可能是 Tomcat 行为。
标签: spring tomcat servlets spring-mvc servlet-3.0