news 2026/4/15 15:12:13

异步校验工具 awaitility

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
异步校验工具 awaitility

1.什么是awaitility ?

Awaitility 是一个用于 Java 的小型领域特定语言(DSL),主要用于简化和管理异步操作的同步问题。它的主要作用包括:

  1. 等待异步操作完成:在测试异步代码时,Awaitility 可以帮助你等待某个条件变为真,而不需要使用复杂的线程管理或轮询机制。

  2. 提高测试的可读性:通过使用流畅的 API,Awaitility 使得测试代码更易于阅读和理解。

  3. 减少测试中的线程问题:避免在测试中显式地使用Thread.sleep(),从而减少不必要的等待时间和线程问题。

  4. 灵活的超时和轮询间隔:允许你设置自定义的超时时间和轮询间隔,以便更好地控制等待条件的检查频率。

总之,Awaitility 使得在测试异步操作时更加简单和直观,特别是在需要等待某个条件满足的情况下。

2.代码工程

实验目的

一个使用 Awaitility 的简单示例,演示如何等待异步操作完成。假设我们有一个异步任务,该任务在后台线程中更新一个标志,我们希望在测试中等待这个标志变为true

pom.xml

<?xml version="1.0" encoding="UTF-8"?><project xmlns="http://maven.apache.org/POM/4.0.0"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"><parent><artifactId>Java-demo</artifactId><groupId>com.et</groupId><version>1.0-SNAPSHOT</version></parent><modelVersion>4.0.0</modelVersion><artifactId>Awaitility</artifactId><properties><maven.compiler.source>17</maven.compiler.source><maven.compiler.target>17</maven.compiler.target></properties><dependencies><!-- Awaitility dependency --><dependency><groupId>org.awaitility</groupId><artifactId>awaitility</artifactId><version>4.2.0</version></dependency><!-- JUnit dependency for testing --><dependency><groupId>org.junit.jupiter</groupId><artifactId>junit-jupiter</artifactId><version>5.8.2</version><scope>test</scope></dependency></dependencies></project>

AwaitilityExample

  1. 异步任务:startAsyncTask方法启动一个异步任务,该任务在 5秒后将flag设置为true

  2. Awaitility 使用:在main方法中,我们使用 Awaitility 的await()方法来等待flag变为true。我们设置了一个最大等待时间为 5 秒。

  3. 条件检查:until(example::isFlag)表示我们等待example.isFlag()返回true

ackage com.et;import org.awaitility.Awaitility;import java.util.concurrent.TimeUnit;import java.util.concurrent.ExecutorService;import java.util.concurrent.Executors;public class AwaitilityExample {private volatile boolean flag = false;public void startAsyncTask() {ExecutorService executor = Executors.newSingleThreadExecutor();executor.submit(() -> {try {// mock asyncThread.sleep(5000);flag = true;} catch (InterruptedException e) {Thread.currentThread().interrupt();}});executor.shutdown();}public boolean isFlag() {return flag;}public static void main(String[] args) {AwaitilityExample example = new AwaitilityExample();example.startAsyncTask();// use Awaitility to wait flag for trueAwaitility.await().atMost(5, TimeUnit.SECONDS).until(example::isFlag);System.out.println("Flag is now true!");}}

以上只是一些关键代码,所有代码请参见下面代码仓库

代码仓库

  • https://github.com/Harries/Java-demo(awaitility )

3.测试代码

3-1.默认等待时间

await().until(Callable conditionEvaluator) 最多等待 10s 直到 conditionEvaluator 满足条件,否则 ConditionTimeoutException。

public void testAsynchronousNormal() {AwaitilityExample example = new AwaitilityExample();example.startAsyncTask();try {// Default timeout is 10 seconds. If the condition is not met within this period, a ConditionTimeoutException is thrownAwaitility.await().until(new Callable<Boolean>() {@Overridepublic Boolean call() throws Exception {return example.isFlag();}});} catch (Exception e) {Assertions.fail("Run exception: " + e.getMessage() + ", error: " + e.getStackTrace()[0].toString());}}

3-2.最多等待

await().atMost() 设置最多等待时间,如果在这时间内条件还不满足,将抛出 ConditionTimeoutException。

@Testpublic void testAsynchronousAtMost() {AwaitilityExample example = new AwaitilityExample();example.startAsyncTask();try {// Specify a timeout of 3 seconds. If the condition is not met within this period, a ConditionTimeoutException is thrownAwaitility.await().atMost(3, SECONDS).until(new Callable<Boolean>() {@Overridepublic Boolean call() throws Exception {return example.isFlag();}});} catch (Exception e) {Assertions.fail("Run exception: " + e.getMessage() + ", error: " + e.getStackTrace()[0].toString());}}

3-3.至少等待

await().atLeast() 设置至少等待时间;多个条件时候用 and() 连接。

@Testpublic void testAsynchronousAtLeast() {AwaitilityExample example = new AwaitilityExample();example.startAsyncTask();try {// Specify at least 1 second and at most 3 seconds. If the condition is not met within this period, a ConditionTimeoutException is thrownAwaitility.await().atLeast(1, SECONDS).and().atMost(3, SECONDS).until(new Callable<Boolean>() {@Overridepublic Boolean call() throws Exception {return example.isFlag();}});} catch (Exception e) {Assertions.fail("Run exception: " + e.getMessage() + ", error: " + e.getStackTrace()[0].toString());}}

3-4.轮询

with().pollInterval(ONE_HUNDRED_MILLISECONDS).and().with().pollDelay(50, MILLISECONDS) that is conditions are checked after 50ms then 50ms+100ms。

@Testpublic void testAsynchronousPoll() {AwaitilityExample example = new AwaitilityExample();example.startAsyncTask();try {// Polling query, pollInterval specifies how often to poll, pollDelay specifies the delay between each pollAwaitility.with().pollInterval(ONE_HUNDRED_MILLISECONDS).and().with().pollDelay(50, MILLISECONDS).await("count is greater 3").until(new Callable<Boolean>() {@Overridepublic Boolean call() throws Exception {return example.isFlag();}});} catch (Exception e) {Assertions.fail("Run exception: " + e.getMessage() + ", error: " + e.getStackTrace()[0].toString());}}

3-5.Fibonacci 轮询

with().pollInterval(fibonacci(SECONDS)) 非线性轮询,按照 fibonacci 数轮询。

@Testpublic void testAsynchronousFibonacciPoll() {AwaitilityExample example = new AwaitilityExample();example.startAsyncTask();try {// Use Fibonacci numbers as the interval: 1, 1, 2, 3, 5, 8,..., default unit is millisecondsAwaitility.with().pollInterval(fibonacci(SECONDS)).await("count is greater 3").until(new Callable<Boolean>() {@Overridepublic Boolean call() throws Exception {return example.isFlag();}});} catch (Exception e) {Assertions.fail("Run exception: " + e.getMessage() + ", error: " + e.getStackTrace()[0].toString());}}

4.引用

  • https://github.com/awaitility/awaitility

  • https://www.liuhaihua.cn/archives/711844.html

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/5 16:29:24

PyBlueZ完整指南:快速掌握Python蓝牙开发核心技术

PyBlueZ完整指南&#xff1a;快速掌握Python蓝牙开发核心技术 【免费下载链接】pybluez Bluetooth Python extension module 项目地址: https://gitcode.com/gh_mirrors/py/pybluez PyBlueZ是一个强大的Python蓝牙扩展模块&#xff0c;让开发者能够轻松访问主机的蓝牙资…

作者头像 李华
网站建设 2026/4/8 15:38:43

如何做一个成功的品牌网站?这4点很关键!

网站建设市场的快速开展&#xff0c;不同层次需求的划分愈加的透明。有高端定制网站建设&#xff0c;也有一般网站建设&#xff0c;随着互联网的持续发展&#xff0c;网站建设也会逐步走向专业化、定制化&#xff0c;为人呈现一种专业性的视觉体会。根据不同的需求提供不同的建…

作者头像 李华
网站建设 2026/4/15 8:25:18

抖音去水印工具完全指南:3分钟批量下载用户全作品

抖音去水印工具完全指南&#xff1a;3分钟批量下载用户全作品 【免费下载链接】TikTokDownload 抖音去水印批量下载用户主页作品、喜欢、收藏、图文、音频 项目地址: https://gitcode.com/gh_mirrors/ti/TikTokDownload 还在为抖音视频水印烦恼&#xff1f;想要批量保存…

作者头像 李华
网站建设 2026/4/14 11:40:01

35_Spring AI 干货笔记之 Google GenAI 文本嵌入

一、Google GenAI 文本嵌入 Google GenAI 嵌入 API 通过 Gemini 开发者 API 或 Vertex AI 使用 Google 的嵌入模型提供文本嵌入生成功能。本文档介绍如何使用 Google GenAI 文本嵌入 API 创建文本嵌入。 Google GenAI 文本嵌入 API 使用密集向量表示。与稀疏向量&#xff08;…

作者头像 李华
网站建设 2026/4/15 4:27:06

FIRE之旅 财务计算器:实时交互式建模与前端性能工程

技术实践观察地址&#xff1a; FIRE之旅 财务计算器 摘要&#xff1a; 现代 Web 金融工具的核心挑战在于实现实时交互式建模&#xff08;Real-Time Interactive Modeling&#xff09;&#xff0c;即在用户调整参数时&#xff0c;能够瞬时、无延迟地反馈复杂的计算结果。本文将…

作者头像 李华
网站建设 2026/4/14 4:27:28

一篇论文,让我哭了三天三夜!

虎贲等考AI智能写作&#xff1a;https://www.aihbdk.com/泪水的重量&#xff1a;当一篇论文成为人生的坩埚第一章 那些在学术黑暗中独自流泪的时刻凌晨三点&#xff0c;图书馆的灯光冷白如霜。你盯着屏幕上那篇仿佛永远无法完成的论文&#xff0c;眼泪无声地滑落&#xff0c;在…

作者头像 李华