java - 子请求的 CompletableFuture

标签 java java-8 completable-future asynchttpclient

我正在尝试了解 Java 8 中的 CompletableFuture。作为其中的一部分,我正在尝试进行一些 REST 调用以巩固我的理解。我正在使用这个库进行 REST 调用:https://github.com/AsyncHttpClient/async-http-client .

请注意,此库会为 GET 调用返回一个 Response 对象。

以下是我正在尝试做的事情:

  1. 调用这个给出用户列表的 URL:https://jsonplaceholder.typicode.com/users
  2. 使用 GSON 将响应转换为用户对象列表。
  3. 遍历列表中的每个用户对象,获取用户 ID,然后从以下 URL 获取用户发布的帖子列表:https://jsonplaceholder.typicode.com/posts?userId=1
  4. 使用 GSON 将每个帖子响应转换为 Post 对象。
  5. 构建 UserPost 对象的集合,每个对象都有一个 User 对象和用户发布的帖子列表。

    public class UserPosts {
    
    private final User user;
    private final List<Post> posts;
    
    public UserPosts(User user, List<Post> posts) {
        this.user = user;
        this.posts = posts;
    }
    
    @Override
    public String toString() {
        return "user = " + this.user + " \n" + "post = " + posts+ " \n \n";
    }
    

我目前的实现方式如下:

package com.CompletableFuture;

import java.util.List;
import java.util.Optional;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.Future;
import java.util.function.Function;
import java.util.stream.Collectors;

import org.asynchttpclient.Response;

import com.http.HttpResponse;
import com.http.HttpUtil;
import com.model.Post;
import com.model.User;
import com.model.UserPosts;

/**
 * Created by vm on 8/20/18.
 */

class UserPostResponse {
    private final User user;
    private final Future<Response> postResponse;

    UserPostResponse(User user, Future<Response> postResponse) {
        this.user = user;
        this.postResponse = postResponse;
    }

    public User getUser() {
        return user;
    }

    public Future<Response> getPostResponse() {
        return postResponse;
    }
}

public class HttpCompletableFuture extends HttpResponse {
    private Function<Future<Response>, List<User>> userResponseToObject = user -> {
        try {
            return super.convertResponseToUser(Optional.of(user.get().getResponseBody())).get();
        } catch (Exception e) {
            e.printStackTrace();
            return null;
        }
    };
    private Function<Future<Response>, List<Post>> postResponseToObject = post -> {
        try {
            return super.convertResponseToPost(Optional.of(post.get().getResponseBody())).get();
        } catch (Exception e) {
            e.printStackTrace();
            return null;
        }
    };

    private Function<UserPostResponse, UserPosts> buildUserPosts = (userPostResponse) -> {
        try {
            return new UserPosts(userPostResponse.getUser(), postResponseToObject.apply(userPostResponse.getPostResponse()));
        } catch (Exception e) {
            e.printStackTrace();
            return null;
        }
    };

    private Function<User, UserPostResponse> getPostResponseForUser = user -> {
        Future<Response> resp = super.getPostsForUser(user.getId());
        return new UserPostResponse(user, resp);
    };

    public HttpCompletableFuture() {
        super(HttpUtil.getInstance());
    }

    public List<UserPosts> getUserPosts() {

        try {
            CompletableFuture<List<UserPosts>> usersFuture = CompletableFuture
                    .supplyAsync(() -> super.getUsers())
                    .thenApply(userResponseToObject)
                    .thenApply((List<User> users)-> users.stream().map(getPostResponseForUser).collect(Collectors.toList()))
                    .thenApply((List<UserPostResponse> userPostResponses ) -> userPostResponses.stream().map(buildUserPosts).collect(Collectors.toList()));

            List<UserPosts> users = usersFuture.get();
            System.out.println(users);
            return users;
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }

}

但是,我不确定我这样做的方式是否正确。更具体地说,在 userResponseToObjectpostResponseToObject 函数中,我在 Future 上调用 get() 方法,这将是阻塞的。

有没有更好的方法来实现这个?

最佳答案

如果您打算使用 CompletableFuture,您应该使用 async-http-client 库中的 ListenableFutureListenableFuture 可以转换为 CompletableFuture

使用 CompletableFuture 的优点是您可以编写处理 Response 对象的逻辑,而无需了解任何有关 futures 或线程的知识。假设你写了以下 4 个方法。 2 发出请求和 2 解析响应:

ListenableFuture<Response> requestUsers() {

}

ListenableFuture<Response> requestPosts(User u) {

}

List<User> parseUsers(Response r) {

}

List<UserPost> parseUserPosts(Response r, User u) {

}

现在我们可以编写一个非阻塞方法来检索给定用户的帖子:

CompletableFuture<List<UserPost>> userPosts(User u) {
    return requestPosts(u)
        .toCompletableFuture()
        .thenApply(r -> parseUserPosts(r, u));
}

以及为所有用户读取所有帖子的阻止方法:

List<UserPost> getAllPosts() {
    // issue all requests
    List<CompletableFuture<List<UserPost>>> postFutures = requestUsers()
            .toCompletableFuture()
            .thenApply(userRequest -> parseUsers(userRequest)
                    .stream()
                    .map(this::userPosts)
                    .collect(toList())
            ).join();


    // collect the results
    return postFutures.stream()
            .map(CompletableFuture::join)
            .flatMap(List::stream)
            .collect(toList());
}

关于java - 子请求的 CompletableFuture,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51954848/

相关文章:

java - 抽象类的实例化

java - 返回 CompletableFuture 的 Spring Transactional 方法

multithreading - 在 supplyAsync 阻塞主线程后使用 thenAccept

java - CompletableFuture.allOf() 在个人 future 之后没有完成

java - HttpURLConnection 和没有互联网

java - 设置 Java OAuth 提供程序

java - 在流过滤器中包含 IgnoreCase 来计算字符串列表中某个特定单词的出现次数

java - SimpleThreadPool 的 Quartz ClassNotFoundException

java - maven basedir - 用前斜杠替换反斜杠

java - 类型转换和方法重载