java - 使用 Apache Async Http 客户端进行异步响应流

标签 java apache-httpclient-4.x apache-httpcomponents apache-httpasyncclient

我正在使用 apache 异步 http 客户端从 azure 存储传输对象。

我只需要返回与流关联的 HttpResponse 对象。我的客户实际上必须从该流中读取数据才能在本地存储文件。

因此 Apache 异步客户端使用 BasicAsyncResponseConsumer,它实际上在调用完成的回调之前将整个文件缓冲在本地内存中。

我正在尝试创建自己的 AbstractAsyncResponseConsumer 实现,以便可以流式传输响应正文,而不是首先实际存储它,但到目前为止尚未成功。

这是最基本的 cosumer 类供引用 ->

public class MyConsumer extends` AbstractAsyncResponseConsumer<HttpResponse> {
@Override
protected void onResponseReceived(HttpResponse response) throws HttpException, IOException {

}

@Override
protected void onContentReceived(ContentDecoder decoder, IOControl ioctrl) throws IOException {

}

@Override
protected void onEntityEnclosed(HttpEntity entity, ContentType contentType) throws IOException {

}

@Override
protected HttpResponse buildResult(HttpContext context) throws Exception {
    return null;
}

@Override
protected void releaseResources() {

}

}

这是发送请求并返回响应的代码 ->

public void getFile(HttpRequestBase request) {

    MyConsumer myConsumer = new MyConsumer();
    HttpAsyncRequestProducer producer = 
    HttpAsyncMethods.create(request);
    CompletableFuture<HttpResponse> future = new CompletableFuture<>();
    return Future<HttpResponse> responseFuture = 
    httpclient.execute(producer,myConsumer,                                                                                                                   
    new FutureCallback<HttpResponse>() {
      @Override
      public void completed(HttpResponse result) {
     //This is called only when all the response body has been read
     //future.complete(Result)

      }
      @Override                                                                      
      public void failed(Exception ex) {
      }
      @Override
      public void cancelled() {                                                                       
      }
   });

return future;

 }

我将向我的客户返回 HttpResponse 对象的 CompletableFuture。

他们不应该等待我的 http 客户端首先读取本地缓冲区中的所有响应正文。

理想情况下,它们应该直接从响应对象中提供的流开始复制。

我应该在消费者的实现中添加什么才能获得所需的结果?

最佳答案

我不知道您是否仍然遇到这个问题,但如果您想要的是一个实际流式传输数据的 InputStream,那么您将需要使用 Apache HttpClient 的阻塞版本。

Java 的内置 InputStreamOutputStream 本质上是阻塞的,因此返回 InputStreamCompletableFuture 本质上是失败的目的。 BasicAsyncResponseConsumer 在内存中缓冲整个响应实际上是正确的做法,因为这是使其真正非阻塞的唯一方法。

您可以查看的另一个选项是HttpAsyncMethods.createZeroCopyConsumer。它的作用是以完全非阻塞的方式将内容存储到文件中。 这是一个例子:

        try (CloseableHttpAsyncClient client = HttpAsyncClients.createDefault()) {
            client.start();
            final CompletableFuture<HttpResponse> cf = new CompletableFuture<>();
            client.execute(
                    HttpAsyncMethods.createGet("https://example.com"),
                    HttpAsyncMethods.createZeroCopyConsumer(new File("foo.html")),
                    new FutureCallback<HttpResponse>() {
                        @Override
                        public void completed(HttpResponse result) {
                            cf.complete(result);
                        }
                        @Override
                        public void failed(Exception ex) {
                            cf.completeExceptionally(ex);
                        }
                        @Override
                        public void cancelled() {
                            cf.cancel(true);
                        }
                    });
            // When cf completes, the file will be ready.
            // The InputStream inside the HttpResponse will be the FileInputStream of the created file.
        }

关于java - 使用 Apache Async Http 客户端进行异步响应流,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50294297/

相关文章:

java - 如何在apache httpclient中获取客户端的 session ID

java - 使用 Http.Core 与 Http.Client 4 对话获取 ConnectionClosedException?

java - 使用 Java 进行跨平台作业调度

java - BeanFactory 未初始化或已关闭 - 在通过 ApplicationContext 访问 bean 之前调用 'refresh'

java - 退出 HttpClient session

java - 不支持的记录版本 SSLv2Hello 使用 CloseableHttpClient

java - HTTP post/API 请求在 bash 上从 CURL 发送时有效,从 Apache http 发送失败

java - Spring Data Elasticsearch - 找不到能够从类型 [java.lang.Long] 转换为类型 [java.time.Instant] 的转换器

java - 可扩展 ListView 上子项的 Intent

java - 如何修复 (SSLException) 收到致命警报 : protocol_version in Java