Playwright高级技巧:自定义选择器与定位器

在日常的Web自动化测试中,我们都遇到过这样的场景:页面上那些没有规范属性、动态生成的元素,让编写稳定的选择器变成了一场噩梦。上周我就花了整整一个下午,只为了定位一个不断变换class名的下拉菜单——这种情况在如今的单页应用中太常见了。

如果你也厌倦了脆弱的CSS选择器,那么自定义选择器与定位器将是你的解放工具。Playwright在这方面提供的灵活性,能让你的测试代码从“勉强能用”变成“坚如磐石”。

为什么我们需要自定义选择器?

先看看这个典型的痛点场景:你正在测试一个React应用,发现页面上的按钮是这么写的:

<button class="bg-blue-500 hover:bg-blue-700 px-4 py-2 rounded-lg"> 提交 </button>

用常规的CSS选择器,你可能会写:

await page.click('button.bg-blue-500');

但问题来了:如果UI设计师调整了样式,把bg-blue-500改成bg-blue-600,你的测试就挂了。更糟糕的是,在大型项目中,这种样式类名变动几乎无法避免。

自定义选择器:定义自己的定位策略

Playwright允许你注册自定义选择器引擎,这有点像定义自己的定位“方言”。让我通过一个实际例子来演示。

假设我们有一个自定义数据属性data-testid,这是目前比较流行的做法:

// 注册一个自定义选择器引擎 await page.locator.register('testId', { // 这个引擎会在浏览器端执行 create(root, selector) { return root.querySelector(`[data-testid="${selector}"]`); }, // 支持查询多个元素 queryAll(root, selector) { return root.querySelectorAll(`[data-testid="${selector}"]`); } }); // 使用方式简洁明了 const submitButton = page.locator('testId=submit-button'); await submitButton.click();

现在,即使按钮的class、结构甚至标签类型改变,只要data-testid="submit-button"保持不变,你的测试就能正常运行。

更复杂的自定义定位器

有时候,简单的属性选择器还不够。考虑一个常见的场景:在一个表格行中,需要找到包含特定文本的单元格所在的行。

// 创建一个定位特定表格行的定位器 function rowWithCellText(text) { return page.locator('tr').filter({ has: page.locator('td', { hasText: text }) }); } // 使用示例:找到包含“张三”的行,然后点击该行的编辑按钮 const targetRow = rowWithCellText('张三'); await targetRow.locator('.edit-btn').click();

这种方法的美妙之处在于它的可读性——代码几乎就是在描述“找到包含‘张三’的行”。

组合定位器:构建复杂查询链

Playwright定位器的真正强大之处在于它们的组合能力。想象一下这个需求:在一个购物车页面,找到第一个数量大于2的商品,然后将其删除。

// 定义可重用的定位器组件 const cartItems = page.locator('.cart-item'); const quantityGreaterThan = (min) => page.locator('.quantity').filter({ hasText: (text) => parseInt(text) > min }); // 组合使用 const targetItem = cartItems .filter({ has: quantityGreaterThan(2) }) .first(); await targetItem.locator('.remove-btn').click();

这种声明式的写法不仅清晰,而且维护起来也容易得多。

处理动态内容和影子DOM

现代Web组件经常使用影子DOM,这给自动化测试带来了额外的挑战。别担心,Playwright也能处理:

// 自定义选择器,穿透影子DOM查找元素 await page.locator.register('shadowId', { create(root, selector) { // 递归查找影子DOM function findInShadow(node, targetId) { if (node.shadowRoot) { const found = node.shadowRoot.querySelector(`[data-id="${targetId}"]`); if (found) return found; // 继续在影子DOM内部查找 for (const child of node.shadowRoot.children) { const result = findInShadow(child, targetId); if (result) return result; } } returnnull; } return findInShadow(root, selector); }, queryAll(root, selector) { const results = []; function findAllInShadow(node, targetId) { if (node.shadowRoot) { const found = node.shadowRoot.querySelectorAll(`[data-id="${targetId}"]`); results.push(...found); for (const child of node.shadowRoot.children) { findAllInShadow(child, targetId); } } } findAllInShadow(root, selector); return results; } });

实际项目中的最佳实践

经过多个项目的实践,我总结出了一些经验:

统一的选择器策略

// selector-utils.js exportconst Selectors = { byTestId: (id) =>`[data-test="${id}"]`, byAriaLabel: (label) =>`[aria-label="${label}"]`, byPartialText: (text) =>`text=${text}`, // 组合定位器 rowByCellText: (tableSelector, text) => page.locator(`${tableSelector} tr`).filter({ has: page.locator('td', { hasText: text }) }) };

等待策略封装

async function waitForLocator(locator, options = {}) { const { timeout = 10000, state = 'visible' } = options; try { await locator.waitFor({ state, timeout }); return locator; } catch (error) { // 添加更有用的错误信息 const html = await page.evaluate(() =>document.documentElement.outerHTML); console.error(`定位器 ${locator} 查找失败,当前页面HTML片段:`, html.substring(0, 1000)); throw error; } }

页面对象模式中的应用

class LoginPage { constructor(page) { this.page = page; } // 使用自定义定位器 get usernameInput() { returnthis.page.locator('testId=username-input'); } get passwordInput() { returnthis.page.locator(this.page.locator.register('byLabel', { create(root, selector) { const label = Array.from(root.querySelectorAll('label')) .find(l => l.textContent.includes(selector)); return label ? root.querySelector(`#${label.getAttribute('for')}`) : null; } })); } async login(username, password) { awaitthis.usernameInput.fill(username); awaitthis.passwordInput.fill(password); awaitthis.page.locator('testId=login-btn').click(); } }

调试技巧

当自定义选择器不工作时,这些调试方法很有帮助:

// 1. 查看定位器匹配的元素数量 const count = await page.locator('your-selector').count(); console.log(`找到 ${count} 个元素`); // 2. 高亮显示匹配的元素 await page.locator('your-selector').highlight(); // 3. 获取匹配元素的详细信息 const elements = await page.locator('your-selector').elementHandles(); for (const [index, element] of elements.entries()) { const tagName = await element.evaluate(el => el.tagName); const text = await element.textContent(); console.log(`元素 ${index}: ${tagName}, 文本: "${text}"`); }

自定义选择器和定位器不是银弹,但它们确实是解决复杂定位问题的强大工具。关键是要找到适合你项目的平衡点——不要过度设计,但也要避免过于脆弱的选择器。

我建议从简单的自定义选择器开始,比如基于data-testid的定位。当遇到更复杂场景时,再逐步引入更高级的技巧。记住,好的定位器应该像好代码一样:意图清晰、易于维护,并且足够健壮以应对变化。

真正的高手不是能写出最复杂的选择器,而是能用最简单的方式解决最棘手的定位问题。希望这些技巧能帮你写出更稳定、更可读的自动化测试代码。

推荐阅读

黑盒测试方法—等价类划分法

大学毕业后转行软件测试我后悔了

软件测试 | 测试开发 | Android动态权限详解

软件测试的测试方法及测试流程

软件测试 | 测试开发 | Android App 保活服务的配置与禁用

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/1156120.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

UE5 如何显示蓝图运行流程

运行UE5蓝图的时候会显示运行时候的步骤&#xff0c;方便调试&#xff0c;具体开启方法如下&#xff1a; 1、打开蓝图编辑器 2、将项目点击Play运行起来 3、这时候这里是显示未选中调试对象 4、下拉选择要调试的对象&#xff08;如果没有下拉选项&#xff0c;确定游戏页面中…

如何构建FunASR的本地语音识别服务

FunASR 简介 FunASR 是阿里巴巴达摩院开源的高性能语音识别工具包&#xff0c;支持离线识别和实时流式识别两种模式。其核心特点包括&#xff1a; 支持多种语音任务&#xff1a;ASR&#xff08;自动语音识别&#xff09;、VAD&#xff08;语音活动检测&#xff09;、标点恢复…

「测试面试官手记」海投三个月零面试,一招拿到了心仪Offer!

真正的机会&#xff0c;从来不是大海捞针 海投简历&#xff0c;可能是这个时代求职者最大的自我安慰。 作为一名在测试行业摸爬滚打十多年的“老兵”&#xff0c;我见过太多同行陷入同一种困境&#xff1a;每天在招聘平台一键投出几十份简历&#xff0c;结果要么石沉大海&#…

给定一个二叉树,求其最近公共祖先

二叉树最近公共祖先(LCA)问题全解析:从理论到实践的完美指南 关键词 二叉树, 最近公共祖先, LCA算法, 树遍历, 递归, 数据结构, 算法优化 摘要 最近公共祖先(Lowest Common Ancestor, LCA)问题是二叉树操作中的经典问题,在计算机科学领域有着广泛的应用。本文将带领读者深…

Arduino下载安装教程:板卡支持包添加方法

Arduino板卡支持包怎么加&#xff1f;一文搞懂BSP背后的硬核逻辑 你是不是也遇到过这种情况&#xff1a;兴冲冲地下载安装好Arduino IDE&#xff0c;连上开发板&#xff0c;结果一编译就报错“找不到WiFi.h”或者“unknown board”&#xff1f;别急——这根本不是你的代码有问…

图网络的度矩阵D/邻接矩阵A/拉普拉斯矩阵L以及图中的节点如何各自保存更新节点特征

在开始前&#xff0c;我们明确几个概念度矩阵D/邻接矩阵A/拉普拉斯矩阵L分别是做什么的&#xff1f; 度矩阵D&#xff1a;描述一个节点能连接多少其他节点&#xff1b;邻接矩阵A: 描述一个节点具体和其他哪个节点连接&#xff1b;拉普拉斯矩阵L&#xff1a;LD-A描述一个节点的特…

车载电子PCB工艺选型要求:项目应用解析

车载电子PCB工艺选型实战指南&#xff1a;从设计到可靠的工程闭环为什么一块车规级PCB不能“照搬”消费类经验&#xff1f;你有没有遇到过这样的情况&#xff1a;同一块电路板&#xff0c;用在工控设备上稳定运行三年&#xff0c;放到发动机舱里却三个月就出现通信中断&#xf…

自动驾驶场景下的Android HMI开发:资深工程师职位深度解析

上海翰格企业管理咨询有限公司 Android资深开发工程师 职位信息 岗位描述:职位描述 1、基于自动驾驶场景需求,开发Android平台上的HMI应用程序,为用户提供友好直观的交互体验 2、整合地图、导航、传感器等系统模块,确保不同数据流在Android HMI系统中的无缝交互与显示 3、实…

XML处理:提取唯一ID的XSLT优化

在处理XML数据时,如何高效地提取唯一标识符(ID)是一项常见的挑战。今天,我们将探讨如何利用XSLT(Extensible Stylesheet Language Transformations)来实现这一目标。特别是,我们将聚焦于解决一个具体问题:从一个包含多个重复ID的XML文档中提取并统计唯一ID的数量。 问…

揭秘大模型 “胡说八道”:幻觉产生的底层原理与规避逻辑

1. 引言&#xff1a;大模型的“幻觉陷阱”离我们有多近当你向大模型询问“爱因斯坦发明了电灯吗”&#xff0c;它可能一本正经地告诉你“是的&#xff0c;爱因斯坦在1879年发明了电灯&#xff0c;这一发明改变了人类的照明方式”&#xff1b;当你让它撰写一篇关于环境治理的论文…

从文本到图像:多模态大模型跨域理解的核心技术原理

1. 引言&#xff1a;为什么“文本变图像”是AI跨域理解的关键突破在AI发展的早期&#xff0c;大多数模型都只能处理单一类型的信息&#xff1a;有的模型只能“读懂”文字&#xff0c;比如智能客服机器人&#xff1b;有的模型只能“看懂”图像&#xff0c;比如人脸识别系统。这种…

C语言中的逻辑与运算误区

在C语言编程中,逻辑运算符的理解和使用是每个程序员必须掌握的基本技能。然而,在实际编程中,很多初学者(甚至是一些经验丰富的程序员)可能会因为一些细微的误解而陷入困惑。今天,我们通过一个具体的例子来深入探讨C语言中的逻辑与运算(&&)。 问题背景 假设有一…

通过SMBus读取电源状态寄存器:操作指南

如何用SMBus读取电源状态寄存器&#xff1f;一文讲透原理与实战你有没有遇到过这样的问题&#xff1a;系统突然宕机&#xff0c;日志里却找不到原因&#xff0c;最后怀疑是电源异常&#xff0c;但又无法复现&#xff1f;在服务器、工业控制板或高性能嵌入式设备中&#xff0c;这…

GeoPandas绘图技巧:如何优雅地在地图上标注县城信息

引言 在使用GeoPandas进行地理数据可视化时,如何在同一张地图上叠加多个信息层并保持整洁清晰,是许多数据分析师和开发者面临的挑战。本文将结合实际案例,介绍如何利用GeoPandas的高级功能,实现在地图上标注县城的名称和面积信息。 GeoPandas简介 GeoPandas是Python的一…

别让错招毁了团队:入职背景调查,为企业把好人才第一关

“面试时思路清晰、态度积极&#xff0c;入职后却频频出错&#xff0c;连简历上的核心项目经验都是编造的”——这是HR小林最近的烦心事。一场看似成功的招聘&#xff0c;最终却让团队陷入返工内耗&#xff0c;还得重新开启招聘流程。其实&#xff0c;这类招聘“踩雷”的背后&a…

数据分析:自动计算近五个月平均值

在日常的工作中,处理大量的时间序列数据是常有的事,尤其当这些数据涉及到月度平均值的计算时,手动更新公式不仅繁琐,而且容易出错。今天我们要讨论如何使用Google Sheets的公式来自动计算并显示过去五个月的平均值,避免了手动调整VLOOKUP等公式的麻烦。 问题背景 假设我…

核心要点:如何判断是STLink损坏还是配置错误

如何精准判断STLink是真坏了还是配置翻车&#xff1f;从物理连接到固件调试的全链路排障实战 你有没有经历过这样的时刻&#xff1f; 刚坐下准备烧个程序&#xff0c;打开STM32CubeProgrammer&#xff0c;点“Connect”——结果弹出一个冷冰冰的提示&#xff1a; No ST-LINK…

AWS云从业者认证(AWS Certified Cloud Practitioner)

一、认证介绍AWS云从业者认证(AWS Certified Cloud Practitioner)是亚马逊云科技(AWS)推出的一系列认证考试中最基础&#xff0c;最入门的一门。它特别适合对云计算和AWS平台了解不多的"小白"或非IT行业从业者&#xff0c;是进入云计算领域的敲门砖。二、考试内容与目…

深入浅出:Java邮件发送中的换行问题

在Java编程中,发送电子邮件是一个常见的任务。然而,当我们尝试在邮件内容中插入换行时,可能会遇到一些意想不到的问题。今天,我们将详细探讨在Java中如何正确地在邮件内容中使用换行符,并通过一个具体的实例来解释这些问题。 问题背景 在Java中,字符串中的换行符通常用…

Proteus仿真环境下单片机定时器配置实战案例

在Proteus中玩转定时器&#xff1a;从代码配置到仿真验证的完整实战你有没有过这样的经历&#xff1f;写完一段定时器中断代码&#xff0c;烧进单片机却发现LED不闪、频率不对&#xff0c;甚至程序直接跑飞。反复查寄存器、对晶振、看延时计算……调试半天&#xff0c;最后发现…