我正在使用新的java.net.http
类来处理异步HTTP请求和响应交换,并且我试图找到一种方法使BodySubscriber处理不同的编码类型,例如gzip。
但是,映射BodySubsriber<InputStream>
使得基础流被GZIPInputStream
包装(当在响应标头中找到“ Content-Encoding:gzip”时)会导致挂起。没有例外,只是完全停止了活动。
映射BodySubscriber
的代码如下:
private HttpResponse.BodySubscriber<InputStream> gzippedBodySubscriber(
HttpResponse.ResponseInfo responseInfo) {
return HttpResponse.BodySubscribers.mapping(
HttpResponse.BodySubscribers.ofInputStream(),
this::decodeGzipStream);
}
private InputStream decodeGzipStream(InputStream gzippedStream) {
System.out.println("Entered decodeGzipStream method.");
try {
InputStream decodedStream = new GZIPInputStream(gzippedStream);
System.out.println(
"Created GZIPInputStream to handle response body stream.");
return decodedStream;
} catch (IOException ex) {
System.out.println("IOException occurred while trying to create GZIPInputStream.");
throw new UncheckedIOException(ex);
}
}
接收具有“ gzip”编码的HTTP响应会导致控制台显示以下内容:
输入了EncodedBodyHandler.apply方法。
输入了encodeGzipStream方法。
什么也看不到,因此GZIPInputStream
构造函数调用之后的行将永远不会执行。
有人知道为什么将InputStream
中的BodySubscriber<InputStream>
包装到GZIPInputStream
中的尝试挂起了吗?
注意:未编码(原始文本)HTTP响应正文的等效方法仅包含对BodySubscribers.ofInputStream()
的调用,而没有任何映射,因此可以毫无问题地接收和显示响应。
答案 0 :(得分:2)
这确实是一个错误。我已登录JDK-8217264。我可以建议两个解决方法:
请勿使用BodySubscribers.mapping
-在获取HttpResponse的正文后,将InputStream
转换为GZIPInputStream
:
GZIPInputStream gzin = new GZIPInputStream(resp.getBody());
让映射函数返回Supplier<InputStream>
,注意不要在调用GZIPInputStream
之前创建Supplier::get
static final class ISS implements Supplier<InputStream> {
final InputStream in;
GZIPInputStream gz;
ISS(InputStream in) {
this.in = in;
}
public synchronized InputStream get() {
if (gz == null) {
try {
gz = new GZIPInputStream(in);
} catch (IOException t) {
throw new UncheckedIOException(t);
}
}
return gz;
}
}
答案 1 :(得分:1)
遇到了完全相同的问题。我在BodySubscribers.mapping
方法的Javadoc中尝试了该示例。行为相同,应用程序挂起,没有任何错误。
可能是一个错误,因为这是Javadoc中的官方示例。
public static <W> BodySubscriber<W> asJSON(Class<W> targetType) {
BodySubscriber<InputStream> upstream = BodySubscribers.ofInputStream();
BodySubscriber<W> downstream = BodySubscribers.mapping(
upstream,
(InputStream is) -> {
try (InputStream stream = is) {
ObjectMapper objectMapper = new ObjectMapper();
return objectMapper.readValue(stream, targetType);
} catch (IOException e) {
throw new UncheckedIOException(e);
}
});
return downstream;
} }