通过这里回答的几个例子和问题(主要是 http://www.javaworld.com/javaworld/jw-02-2009/jw-02-servlet3.html?page=3),我希望服务器多次向客户端发送响应而不完成请求。当请求超时时,我会创建另一个,依此类推。
我想避免长时间轮询,因为每次收到响应时我都必须重新创建请求。 (而且这不是servlet 3.0的异步功能所针对的。)
我在服务器端有这个:
@WebServlet(urlPatterns = {"/home"}, name = "async", asyncSupported = true)
public class CometServlet extends HttpServlet {
public void doGet(final HttpServletRequest request, final HttpServletResponse response) throws IOException, ServletException {
AsyncContext ac = request.startAsync(request, response);
HashMap<String, AsyncContext> store = AppContext.getInstance().getStore();
store.put(request.getParameter("id"), ac);
}
}
写入异步上下文的线程。
class MyThread extends Thread {
String id, message;
public MyThread(String id, String message) {
this.id = id;
this.message = message;
}
public void run() {
HashMap<String, AsyncContext> store = AppContext.getInstance().getStore();
AsyncContext ac = store.get(id);
try {
ac.getResponse().getWriter().print(message);
} catch (IOException e) {
e.printStackTrace();
}
}
}
但是当我发出请求时,只有在我拨打ac.complete()
时才会发送数据。没有它请求将永远超时。所以基本上我想在请求完成之前将数据“流式传输”。
只是为了记笔记,我已尝试使用Jetty 8
续作API
,我也尝试打印到OutputStream
而不是PrintWriter
。我还在回复时尝试了flushBuffer()
。同样的事情。
我做错了什么?
客户端就是这样完成的:
var xhr = new XMLHttpRequest();
xhr.open('GET', 'http://localhost:8080/home', true);
xhr.onreadystatechange = function () {
if (xhr.readyState == 3 || xhr.readyState == 4) {
document.getElementById("dynamicContent").innerHTML = xhr.responseText;
}
}
xhr.send(null);
有人至少可以确认服务器方面没问题吗? :)
答案 0 :(得分:8)
您的服务器端和客户端代码确实没问题。 问题实际上是您的浏览器缓冲来自您的Web服务器的 text / plain 响应。 这是您在使用curl时没有看到此问题的原因。
我使用了客户端代码,我能够看到增量响应,仅只有一点变化:
response.setContentType("text/html");
无论大小如何,增量响应都会立即出现。
如果没有这个设置,当我的输出是一个小消息时,它被认为是text / plain并且没有立即显示在客户端。当我不断向客户端响应添加越来越多的内容时,它会累积到缓冲区大小达到大约1024字节,然后整个东西出现在客户端。然而,在那之后,立即出现小增量(不再累积)。
答案 1 :(得分:0)
我知道这有点旧,但你也可以在响应中刷新。