0

0

Cypress测试中JavaScript异步执行与数据持久化实践

DDD

DDD

发布时间:2025-10-15 09:58:01

|

964人浏览过

|

来源于php中文网

原创

Cypress测试中JavaScript异步执行与数据持久化实践

本文深入探讨了cypress测试中常见的javascript异步执行顺序问题及其解决方案。当cypress命令与普通javascript代码混合时,可能导致变量值未按预期更新。文章详细阐述了如何利用`cy.then()`命令确保cypress命令的顺序执行,以及如何通过`cypress.env()`在页面刷新后依然持久化数据,从而解决测试中的竞态条件和数据丢失问题,提升测试的稳定性和可靠性。

理解Cypress命令队列与JavaScript异步执行

在Cypress测试中,一个常见的困惑是JavaScript代码的执行顺序与预期不符。Cypress命令(如cy.get()、cy.invoke())并非立即执行,而是被添加到一个内部的命令队列中。这些命令是异步的,它们会按照添加的顺序依次执行,但其结果的获取和后续处理需要通过.then()回调来完成。而普通的JavaScript语句,如变量赋值和console.log(),则是同步执行的。

考虑以下示例代码,其目标是从UI元素中提取一个数字,并在后续操作中使用它:

const MATCHING_MESSAGE = '[data-cy=matchingMessages]';

let count = 0; // 初始化变量
cy.get(MATCHING_MESSAGE)
  .invoke('text')
  .then((text) => {
    const pattern = /[0-9]+/g;
    count = text.match(pattern).pop(); // 在Cypress命令队列中赋值
    console.log({count1: count}); // 此时count已被更新
  });

// 模拟页面跳转或其他异步操作
// move to next page

console.log({count2: count}); // 这里的count值可能不是期望的
if (count > 0) {
  // input count
} else {
  // input a min number like 5
}

在上述代码中,cy.get(...).then(...)是一个异步操作,它会被推入Cypress命令队列。而console.log({count2: count});和后续的if判断是同步的JavaScript代码,它们会立即执行,而不会等待Cypress命令队列中的count变量被赋值。这导致count2在count1之前输出,且其值为初始值0,从而引发逻辑错误。

解决方案一:使用cy.then()强制执行顺序

为了确保依赖于Cypress命令结果的代码能够在其结果可用后才执行,我们必须将这些代码也纳入Cypress的命令队列中。cy.then()命令正是为此目的而设计的。它允许你在Cypress命令链的任何位置插入普通的JavaScript代码,并确保这些代码会在前面的Cypress命令完成后执行。

立即学习Java免费学习笔记(深入)”;

通过将后续逻辑包裹在cy.then()中,我们可以强制Cypress等待count变量被正确赋值后再执行依赖于它的代码:

const MATCHING_MESSAGE = '[data-cy=matchingMessages]';

let count = null; // 建议初始化为null,以便区分未赋值和实际值为0的情况

cy.get(MATCHING_MESSAGE)
  .invoke('text')
  .then((text) => {
    const pattern = /[0-9]+/g;
    count = text.match(pattern).pop();
    console.log({count1: count});
  });

// 模拟页面跳转或其他异步操作
// move to next page

cy.then(() => {
  // 此处的代码会在Cypress命令队列中,确保在前面的cy.get().then()执行完毕后才运行
  console.log({count2: count});

  // 注意:从UI提取的数字通常是字符串,进行数值比较前需要转换为数字类型
  if (parseInt(count) > 0) {
    // input count
  } else {
    // input a min number like 5
  }
});

在这个修正后的版本中,cy.then()块内的代码现在也是Cypress命令队列的一部分,因此它会等待前面的cy.get().invoke().then()链执行完毕,count变量被正确赋值后,再进行console.log和条件判断。

AIBox 一站式AI创作平台
AIBox 一站式AI创作平台

AIBox365一站式AI创作平台,支持ChatGPT、GPT4、Claue3、Gemini、Midjourney等国内外大模型

下载

解决方案二:使用Cypress.env()持久化数据

上述cy.then()解决了执行顺序问题,但如果// move to next page这个操作导致了页面的完全刷新(例如,通过cy.visit()或点击导致新页面加载),那么之前在JavaScript作用域中定义的count变量将会丢失,因为它属于旧的页面上下文。

为了在页面刷新或测试步骤之间持久化数据,Cypress提供了Cypress.env()API。Cypress.env()允许你在Cypress环境中存储和检索键值对,这些数据会在整个测试运行期间保持不变,即使页面刷新也不会丢失。

以下是如何使用Cypress.env()来持久化count变量的示例:

const MATCHING_MESSAGE = '[data-cy=matchingMessages]';

cy.get(MATCHING_MESSAGE)
  .invoke('text')
  .then((text) => {
    const pattern = /[0-9]+/g;
    const extractedCount = text.match(pattern).pop();
    console.log({count1: extractedCount});
    Cypress.env('count', extractedCount); // 将数据保存到Cypress环境中
  });

// 模拟页面跳转或其他异步操作,即使页面刷新,Cypress.env()中的数据也依然存在
// move to next page

cy.then(() => {
  // 从Cypress环境中恢复数据
  const count = Cypress.env('count');

  console.log({count2: count});

  if (parseInt(count) > 0) {
    // input count
  } else {
    // input a min number like 5
  }
});

通过Cypress.env('count', extractedCount),我们将提取到的数字存储在Cypress环境中。即使页面刷新,我们也可以在后续的cy.then()块中通过Cypress.env('count')安全地检索到这个值。

总结与注意事项

  1. 理解异步性: Cypress命令是异步的,它们被添加到命令队列中。普通JavaScript代码是同步执行的。混合使用时,务必注意执行顺序。
  2. 利用cy.then(): 当你需要对Cypress命令的结果进行后续的JavaScript处理或逻辑判断时,请将这些代码包裹在cy.then()回调中,以确保正确的执行顺序。
  3. 数据持久化: 如果你的测试流程涉及页面刷新,且需要在刷新前后共享数据,请使用Cypress.env()来存储和检索这些数据,以避免变量丢失。
  4. 类型转换: 从UI元素中提取的文本通常是字符串。如果需要进行数值比较或算术运算,请记得使用parseInt()或parseFloat()进行类型转换。
  5. 初始化变量: 声明变量时,可以将其初始化为null或一个明显无效的值,这样在调试时可以更容易地判断变量是否已被正确赋值。

掌握这些概念和技巧,将有助于你编写更健壮、更可靠的Cypress测试,有效避免因异步执行和数据丢失导致的测试失败。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

254

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

1089

2024.03.01

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

847

2023.08.22

counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

203

2023.11.20

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

761

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

221

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1569

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

651

2023.11.24

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

26

2026.03.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 6.1万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 3.4万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3.6万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号