当前位置: 首页 > 面试题库 >

使用Spring WebFlux中的webclient在Mono上有条件地重复或重试

郭子航
2023-03-14
问题内容

我想做的是在有webclient的Webflux中的Mono上有条件地重复。情况如下:

我们有一些商务休息服务服务,可返回生成的文档。此文档的生成是由在此之前调用的另一服务触发的。但是,回到正题:文档生成服务需要10到30秒。我们想要做的是:10秒钟后检查是否生成文档(单声道)。如果是这样,一切都很好。如果不是,请在5秒钟后重复(或重试)并检查是否生成了文档。依此类推,直到(最坏的情况)直到30秒后超时。这可能吗?一些(伪)代码

return this.webClient.post().uri(SERVICE_URL)).        
body(BodyInserters.fromObject(docRequest)).retrieve().
bodyToMono(Document.class).
delaySubscription(Duration.ofSeconds(10)).
repeat5TimesWithDynamicTimeDelayUntil(!document.isEmpty()).
subscribe();

格蕾兹·伯纳多


问题答案:

是的,有可能。

Mono 有两个用于重新订阅(因此,重新触发请求)的概念

  • 重试 =重新订阅上游是否完成但有异常
  • 重复 =重新订阅上游是否成功完成

每个概念都有Mono针对不同用例的多种重载方法。寻找retry*repeat*方法。例如,要无延迟重试最大次数,请使用retry(int numRetries)

通过retryWhenrepeatWhen方法支持更复杂的用例,如以下示例所示。

重试时间

要重试单声道是否异常完成最多5次,每次尝试之间间隔5秒:

// From reactor-core >= v3.3.4.RELEASE
import reactor.util.retry.Retry;

this.webClient
        .post()
        .uri(SERVICE_URL)
        .body(BodyInserters.fromValue(docRequest))
        .retrieve()
        .bodyToMono(Document.class)
        .retryWhen(Retry.fixedDelay(5, Duration.ofSeconds(5)))
        .delaySubscription(Duration.ofSeconds(10))

重试构建器支持其他退避策略(例如指数)和其他选项,以完全自定义重试。

请注意,retryWhen(Retry)上面使用的方法已添加到反应堆核心v3.3.4.RELEASE中,并且retryWhen(Function)不建议使用该方法。在reactor-
core v3.3.4.RELEASE之前,您可以使用reactor-
extras
项目中的重试html" target="_blank">功能构建器来创建一个Function传递给retryWhen(Function)

重复时间

如果您需要成功重复一次,请使用.repeatWhen.repeatWhenEmpty代替.retryWhen上面的内容。

使用reactor-
extras
项目中的重复函数构建器来创建重复Function,如下所示:

// From reactor-extras
import reactor.retry.Repeat;

this.webClient
        .post()
        .uri(SERVICE_URL)
        .body(BodyInserters.fromValue(docRequest))
        .retrieve()
        .bodyToMono(Document.class)
        .filter(document -> !document.isEmpty())
        .repeatWhenEmpty(Repeat.onlyIf(repeatContext -> true)
                .exponentialBackoff(Duration.ofSeconds(5), Duration.ofSeconds(10))
                .timeout(Duration.ofSeconds(30)))
        .delaySubscription(Duration.ofSeconds(10))

如果要重新订阅成功或失败,也可以将a .retry*与a 链接在一起.repeat*



 类似资料:
  • 我要做的是用WebClient对Webflux中的单声道进行有条件的重复。情况如下: 我们有一些返回生成文档的业务rest服务。此文档的生成是从在此文档之前调用的另一个服务触发的。但是,回到业务:文档生成服务需要10-30秒。我们要做的是:10秒后检查是否生成了文档(单)。如果是这样,一切都很好。如果没有,请在5秒后重复(或重试)并检查是否生成了文档。以此类推,直到(最坏的情况)30秒后超时。这可

  • 我想从df1中删除某些行。我确实以这种方式编写了条件,并向我显示了要删除的确切行。但是,当我尝试对此数据应用drop时,它不起作用: 当我使用 我可以看到我要删除的数据,这意味着代码有效。然而,当我尝试删除这些行时,它不起作用 我也试过了 但它要么表明: 或 如何删除在(待删除)数据框中指定的这些行? 非常感谢。

  • 我正在学习Python,我遇到了一点问题。在我正在学习的课程中看到类似的东西后,我想到了这个简短的脚本。我以前用过“或”和“如果”来表示成功(这里没有太多说明)。出于某种原因,我似乎无法让它工作: 但这很有效: 可能条件不适合这里。但是我试过了,等等。我想有一种方法让它接受猴子或猴子,其他一切都会触发精灵。

  • 我试图使用WebClient实现以下场景。使用RestTemplate很琐碎,但我再也做不到了。

  • 我试图根据从后端系统获得的值,有条件地将一个类赋给一个div。