java - 主线程不等待订阅者完成响应式(Reactive)订阅者的任务

标签 java spring multithreading spring-webflux project-reactor

我在 Spring 中有一个服务,需要使用十种不同的方法获取数据。

我想让这些方法并行执行来执行一些数据库操作并返回到父线程。但父线程应该等待所有响应到来,然后返回响应。

在我当前的方法中,我使用响应式(Reactive)单声道异步执行所有方法,但主线程不等待订阅者方法完成。

以下是我订阅的两个方法

private Mono<BaseResponse> getProfileDetails(long profileId){
        return new Mono<BaseResponse>() {

            @Override
            public void subscribe(Subscriber<? super BaseResponse> s) {
                try {
                    Thread.sleep(5000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                // DB Operation
                System.out.println("Inside getProfileDetails");
                s.onNext(new BaseResponse());
            }
        };
    }

private Mono<Address> getAddressDetails(long profileId){
        return new Mono<Address>() {

            @Override
            public void subscribe(Subscriber<? super Address> s) {
                try {
                    Thread.sleep(5000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                // DB Operation
                System.out.println("Inside getAddressDetails");
                s.onNext(new Address());
            }
        };
    }

下面是我的主要方法

public BaseResponse getDetails(long profileId){
        ExecutorService executors = Executors.newFixedThreadPool(2);

        Mono<BaseResponse> profileDetail = this.getProfileDetails(profileId).subscribeOn(Schedulers.fromExecutor(executors));
        Mono<BaseResponse> addressDetail = this.getAddressDetails(profileId).subscribeOn(Schedulers.fromExecutor(executors));

        List<BaseResponse> list = new ArrayList<>();

        profileDetail.mergeWith(addressDetail)
        .subscribe(consumer -> {
                list.add(consumer);
        });

        System.out.println("list: "+new Gson().toJson(list));
        executors.shutdown();

        return response;
    }

下面是我的输出:

list: []
Inside getProfileDetails
Inside getAddressDetails

我的输出显示主线程没有等待订阅者完成其任务, 那么我该如何处理这种情况呢?

最佳答案

我假设您的 getProfileDetails()getAddressDetails() 方法只是占位符,因为它们没有多大意义。

话虽这么说,如果这是您的整个应用程序,并且您确实只想在完成之前阻止,那么您也可以将当前的 subscribe() 调用更改为 doOnNext (),然后只是 blockLast():

profileDetail.mergeWith(addressDetail)
.doOnNext(consumer -> {
        list.add(consumer);
})
.blockLast();

在响应式(Reactive)应用程序中阻塞响应式(Reactive)线程通常是不明智的,但在这种情况下,您实际上只想在彻底退出之前阻塞 - 所以我在这里看不到太多的缺点。

关于java - 主线程不等待订阅者完成响应式(Reactive)订阅者的任务,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59664460/

相关文章:

java - 如何从 CLI 运行带有外部库的 Java 程序

c++ - 一起使用 std::thread 和 CUDA

c++ - 从多个进程和 SQLITE_BUSY 访问 SQLite 数据库

java - 安卓okhttp库

java - 如果满足条件,则打印数组列表的选定部分

java - Spring /hibernate 。在Web服务中将字符串转换为具有相同时区的日期并插入到SQL中

java - Thymeleaf 条件 img src

spring - JpaRepository : No Transaction in progress

c++ - 在第二个线程中显示图像,OpenCV?

java - ByteArrayOutputStream 到字符串数组