首页 > 解决方案 > 竞争条件,较小的请求优先于较大的 Java 8 异步

问题描述

我为函数 getArtifactDataList() 提供了一个基本 url,然后我得到了一个包含 2 个 InputStream 的列表。InputStreams 的大小应该不同,因为它们是我下载的非常不同的工件。然而它们是一样的。

代码:

    <dependency>
        <groupId>org.glassfish.jersey.core</groupId>
        <artifactId>jersey-client</artifactId>
        <version>2.16</version>
    </dependency>
import org.apache.commons.io.IOUtils;

import javax.ws.rs.client.Client;
import javax.ws.rs.client.ClientBuilder;
import javax.ws.rs.client.Invocation;
import javax.ws.rs.client.WebTarget;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.UriBuilder;
import java.io.InputStream;
import java.net.URL;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;


private static final String DOWNLOAD_1 = "http://some.random/host/artifact.zip";
private static final String DOWNLOAD_2 = "http://random.other/host/artifact2.zip"


public static void main(String[] args) throws Exception {

    List<InputStream> artifactData = getArtifactDataList();

    for(InputStream is : artifactData) {
        System.out.println("Size: "+ IOUtils.toByteArray(is).length);
    }
}

public static final List<InputStream> getArtifactDataList() throws Exception {

    List<InputStream> listData = new ArrayList<>();

    URL downloadUrl = UriBuilder.fromUri(DOWNLOAD_1).buildFromMap(new HashMap<>()).toURL();

    final InputStream data = requestData(downloadUrl); // should be 20MB in size


    URL downloadFontsUrl = UriBuilder.fromUri(DOWNLOAD_2).buildFromMap(new HashMap<>()).toURL();
    final InputStream fontsData = requestData(downloadFontsUrl); // should be 1MB in size

    listData.add(fontsData);
    listData.add(data);

    return listData;
}

public static final InputStream requestData(URL url) {
    Client c = ClientBuilder.newClient();
    WebTarget resource = c.target(url.toString());

    Invocation.Builder request = resource.request();
    request.accept(MediaType.APPLICATION_XML);

    final InputStream data =
            request.get().readEntity(InputStream.class);

    return data;
}

输出

大小:1MB

大小:1MB

似乎 2 个请求中较小的一个(fontsData)以某种方式覆盖了较大的 2 个。我几乎不知道发生了什么。似乎是一个竞争条件,但我不想同步 requestData 函数。是否有其他一些非常简单易用的解决方案?也许是 Java 期货或承诺?在我的情况下如何使用?谁能告诉我这里到底发生了什么?如何解决这个问题?

编辑:此代码旨在说明我的问题。我将流添加到列表中。还添加了网址生成器(我认为这无关紧要)

工作示例

标签: javaasynchronousjava-8callbackjersey-client

解决方案


推荐阅读