为什么Spring DeferredResult在我的应用程序中连续执行?

时间:2017-08-25 20:04:46

标签: java spring spring-mvc tomcat concurrency

我正在尝试构建一个REST Web服务器,其中GET请求是非阻塞的,即使它需要进行一些稍微耗时的调用。

@RestController
public class Endpoint {
    static int callCount = 0;

    @RequestMapping (value = "/endpoints", method = RequestMethod.GET)
    public DeferredResult<Integer> someGetMethod() {
        System.out.println("Mita: GET Called. Count#: " + callCount++);
        DeferredResult<Integer> deferredResult = new DeferredResult<>();
        new Thread( () -> {
            deferredResult.setResult(getSomething());
        }).start();
        System.out.println("Mita: Thread started");
        return deferredResult;
    }

    private int getSomething() {
        try {
            Thread.sleep(5000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        return 100;
    }
}

为了测试,我以大约2秒的间隔向Restlet发送了两个来自Restlet的GET请求。但是,我在spring应用程序的DEBUG级别日志中看到以下内容。

2017-08-26 01:16:38.231 DEBUG 1252 --- [nio-8080-exec-1] o.a.coyote.http11.Http11InputBuffer      : Received [GET /endpoints/ HTTP/1.1
Host: localhost:8080
Connection: keep-alive
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.101 Safari/537.36
Accept: */*
DNT: 1
Accept-Encoding: gzip, deflate, br
Accept-Language: en-US,en;q=0.8,es;q=0.6,zh-CN;q=0.4,zh;q=0.2
 ]
...
...
2017-08-26 01:16:43.399 DEBUG 1252 --- [nio-8080-exec-2] o.a.coyote.http11.Http11InputBuffer      : Received [GET /endpoints/ HTTP/1.1
Host: localhost:8080
Connection: keep-alive
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.101 Safari/537.36
Accept: */*
DNT: 1
Accept-Encoding: gzip, deflate, br
Accept-Language: en-US,en;q=0.8,es;q=0.6,zh-CN;q=0.4,zh;q=0.2

]

注意,两个请求是如何在5秒之后出现的,我称之为睡眠的确切间隔(即使我在第一个请求后约2秒发送了请求)。因此,似乎tomcat正在对传入的请求进行顺序化。如何让tomcat不对请求进行顺序化?或者是我错过了一些非常明显的东西。

1 个答案:

答案 0 :(得分:1)

使用DeferredResult(或它使用的Servlet异步模式)不会阻止客户端(在这种情况下 Restlet )在第一次请求得到响应之前不等待5秒。

调用Spring端点的客户端看起来就像Spring端点根本没有使用异步模式一样。这意味着 Restlet 或Tomcat可能会等到第一个请求完成后再向您的终端发送第二个请求。