public class ListServlet extends HttpServlet { private static final long serialVersionUID = 1L; protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { response.getWriter().println("I am begin output !"); response.getWriter().flush(); //方式一 AsyncContext async = request.startAsync(); new AsyncOutput(async).start(); //方式二 new ThreadOutput(response).start(); response.getWriter().println("I am finash output !"); response.getWriter().flush(); } } class AsyncOutput extends Thread { private AsyncContext async; public AsyncOutput(AsyncContext async) { this.async = async; } public void run() { try { Thread.sleep(3000); async.getResponse().getWriter().println("I was three minutes late !"); async.getResponse().getWriter().flush(); }catch(Exception e) { e.printStackTrace(); } } } class ThreadOutput extends Thread { private HttpServletResponse response; public ThreadOutput(HttpServletResponse response) { this.response = response; } public void run() { try { Thread.sleep(3000); response.getWriter().println("I was three minutes late !"); response.getWriter().flush(); }catch(Exception e) { e.printStackTrace(); } } }
这两种效果是一样啊 既然如此,那还搞什么AsyncContext AsyncContext有什么不一样的地方 以上两种使用方式,有什么区别? 求指点
解答:
AsyncContext不是让你异步输出,而是让你同步输出,但是解放服务器端的线程使用,使用AsyncContext的时候,对于浏览器来说,他们是同步在等待输出的,但是对于服务器端来说,处理此请求的线程并没有卡在那里等待,则是把当前的处理转为线程池处理了,关键就在于线程池,服务器端会起一个线程池去服务那些需要异步处理的请求,而如果你自己每次请求去起一个线程处理的话,这就有可能会耗大量的线程。
你目前对AsyncContext 的使用并不是最佳实践,实际上应该这样使用:
final AsyncContext asyncContext = request.getAsyncContext(); //添加监听器监听异步的执行结果 asyncContext.addListener(new AsyncListener() { @Override public void onComplete(AsyncEvent event) throws IOException { //在这里处理正常结束的逻辑 } @Override public void onTimeout(AsyncEvent event) throws IOException { //在这里处理超时的逻辑 } @Override public void onError(AsyncEvent event) throws IOException { //在这里处理出错的逻辑 } @Override public void onStartAsync(AsyncEvent event) throws IOException { //在这里处理开始异步线程的逻辑 } }); //设置超时的时间,到了时间以后,会回调onTimeout的方法 asyncContext.setTimeout(10000L); //在这里启动,传入一个Runnable对象,服务器会把此Runnable对象放在线程池里面执行 asyncContext.start(new Runnable() { @Override public void run() { //在这里做耗时的操作,如果做完,则调用complete方法通知回调,异步处理结束了 asyncContext.complete(); } });