gpt4 book ai didi

java - 如何将 Mono 变成真正的异步(非响应式!)方法调用?

转载 作者:行者123 更新时间:2023-12-05 08:46:16 29 4
gpt4 key购买 nike

我有一个方法

@Service
public class MyService {
public Mono<Integer> processData() {
... // very long reactive operation
}
}

在正常的程序流程中,我通过 Kafka 事件异步调用此方法。

出于测试目的,我需要将该方法公开为 Web 服务,但该方法应以异步方式公开:仅返回 HTTP 代码 200 OK(“请求已接受”)并继续进行数据处理在后台。

仅调用 Mono#subscribe() 并从 Controller 方法返回是否可以(= 它没有任何不良副作用)?

@RestController
@RequiredArgsConstructor
public class MyController {
private final MyService service;

@GetMapping
public void processData() {
service.processData()
.subscribeOn(Schedulers.boundedElastic())
.subscribe();
}
}

还是这样比较好(这里我被IntelliJ的警告弄糊涂了,可能和https://youtrack.jetbrains.com/issue/IDEA-276018一样?):

public Mono<Void> processData() {
service.processData()
.subscribeOn(Schedulers.boundedElastic())
.subscribe(); // IntelliJ complains "Inappropriate 'subscribe' call" but I think it's a false alarm in my case(?)
return Mono.empty();
}

还是其他一些解决方案?

最佳答案

Is it OK (= doesn't it have any unwanted side effects) just to call Mono#subscribe() and return from the controller method?

有副作用,但您可以接受它们:

  • 这确实是即刻即弃 - 这意味着虽然您永远不会收到成功的通知(大多数人都意识到这一点),但您也永远不会收到失败的通知(很少有人意识到这一点)。
  • 如果进程由于某种原因挂起,发布者将永远不会完成,而您将无从知晓。由于您订阅的是有界弹性线程池,它也会无限期地占用其中一个有限线程。

您可能对第一点没有意见,或者您可能希望将一些错误记录作为副作用以某种方式进一步记录在 react 链中,这样如果出现问题,您至少会有一个内部通知。

对于第二点 - 我建议在您的方法调用上设置一个(慷慨的)超时,这样如果它没有在设定的时间内完成,它至少会被取消,并且不再浪费资源。如果您正在运行异步任务,那么这不是一个大问题,因为它只会消耗一点内存。如果您在弹性调度程序上包装一个阻塞调用,那么情况会更糟,因为您会无限期地占用该线程池中的一个线程。

我还质疑为什么您需要在这里完全使用有界弹性调度程序 - 它用于包装阻塞调用,这似乎不是该用例的基础。 (要明确的是,如果您的服务正在阻塞,那么您绝对应该将其包装在弹性调度程序上——但如果没有,则没有理由这样做。)

最后,这个例子:

public Mono<Void> processData() {
service.processData()
.subscribeOn(Schedulers.boundedElastic())
.subscribe();
return Mono.empty();
}

...是做的一个很好的例子,因为您正在创建一种“冒名顶替者 react 方法”——有人可能非常合理地订阅返回的发布者,认为它会完成当基础发布者完成时,这显然不是这里发生的事情。在这种情况下,使用 void 返回类型并因此不返回任何内容是正确的做法。

关于java - 如何将 Mono 变成真正的异步(非响应式!)方法调用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/70072530/

29 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com