33

我来自 Java SE 背景,我做了一些 servlet 教程并阅读了 Head First JSP & servlet。我现在正在阅读JavaWorld.com关于异步支持的文章,但我不太明白。

什么是异步?Ajax 和 Servlet 异步有什么区别?

PS我有ajax的PHP背景,我知道这个概念,但我没有用java尝试过

4

1 回答 1

84

在传统的 Servlet 模型中,通常是 1 个请求对应 1 个线程。

这些线程通常来自由 Servlet 容器管理的池。Servlet 容器只有在该池中有空闲线程时才能处理新请求。只要您自己的代码忙于处理请求,线程就不是空闲的。

在某些情况下,打破这种模式可能是值得的。发生的情况是请求通过这样的 Servlet 容器托管线程到达 Servlet,然后您的代码要求异步执行。然后您可以从 Servlet 请求返回,容器线程将被释放。

与同步请求处理相反,这不会提交任何响应,也不会关闭连接。相反,您可以将异步上下文交给另一个线程池,该线程池可以获取它,当某个线程有空闲处理它时,为它提供服务并能够写入响应。

一个例子:

@WebServlet(urlPatterns = "/somepath", asyncSupported = true)
public class AsyncServlet extends HttpServlet {

    @EJB
    private AsyncBean asyncBean;

    public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        AsyncContext asyncContext = request.startAsync();

        // The following line will not block and return immediately
        asyncBean.doAsyncStuff(asyncContext);

    } // Shortly after this method has ended, thread will be returned to pool
}

随着AsyncBean被实施为:

@Stateless
public class AsyncBean {

    @Asynchronous
    public void doAsyncStuff(AsyncContext asyncContext) throws IOException {
        asyncContext.getResponse().getWriter().write("test");
    }
}

在上面的代码中,在您从方法返回后不久AsyncServlet#doGet(),Servlet 线程将返回到池中。用于执行的“请求”(任务)AsyncBean#doAsyncStuff()将被放入队列中以供 EJB 线程池获取。

为什么以及何时使用它的答案并不那么简单。如果您只是想保留线程,那么在上述情况下,您将用一个线程池中的一个线程交换另一个线程(在这种情况下是 Servlet 池与 EJB 异步池),净收益不会那么多。你也可以给你的 Servlet 线程池一个额外的线程。

但是,在更高级的场景中,您可以对请求进行更细粒度的管理;将它们分成多个任务,并让线程池为这些任务提供服务。例如,想象一个 10MB 文件的 100 个下载请求,由 10 个线程处理,循环给予每次发送 100KB 到每个请求。

另一个应用程序是需要等待来自外部系统的数据的请求,并且该外部系统能够发送可以中继回请求者的消息。即数据库调用在这里没有意义,因为无论如何您都需要另一个线程等待响应。然后,您将再次将一个线程更改为另一个线程。但是,如果您需要等待收到的电子邮件,那么一个线程可以等待任何电子邮件并将其中继到任何暂停的请求。

于 2012-04-08T20:52:13.583 回答