详解Selenium 强制等待、隐式等待和显式等待

在Selenium中使用等待机制:强制等待、隐式等待和显式等待

在自动化测试中,等待是一个非常重要的部分,因为它可以帮助确保测试脚本在执行操作前,页面元素已经加载完成或者变为可用。Selenium WebDriver 提供了多种等待机制,本文将详细介绍强制等待、隐式等待和显式等待,并给出相应的Java代码示例。


强制等待(Thread.sleep)

概述:

强制等待是通过暂停当前线程一段固定的时间来实现的。这种等待方式不依赖于任何条件,即使目标元素已经加载完成,线程也会继续等待直到指定时间结束。

优点:

  • 简单直接,易于理解和使用。

缺点:

  • 固定等待时间可能导致效率低下。
  • 无法响应元素提前加载,导致不必要的等待。
  • 不推荐在生产环境中使用,只适合调试阶段。

代码示例:

import org.openqa.selenium.WebDriver;
import org.openqa.selenium.chrome.ChromeDriver;public class ForceWaitExample {public static void main(String[] args) {WebDriver driver = new ChromeDriver();driver.get("http://xxx.com");// 强制等待5秒try {Thread.sleep(5000);} catch (InterruptedException e) {e.printStackTrace();}// 继续执行后续操作driver.quit();}
}

隐式等待(Implicit Wait)

概述:

隐式等待(Implicit Wait)是一种全局等待策略,它会在找不到元素时等待一段时间。如果在指定时间内元素仍未出现,WebDriver 会抛出 NoSuchElementException。一旦设置,隐式等待将在整个WebDriver实例的生命周期内都有效。

简单点来理解,隐式等待是对所有的查找都起作用,比如,我在代码的第一行设置了隐式等待,那么后续出现的所有查找相关的代码(不管查找什么,只要和查找相关),就会依次等待。

  • 一次设置,全局有效。
  • 减少由于元素未加载完成而导致的NoSuchElementException

缺点:

  • 对于不同操作可能需要不同的等待时间,隐式等待无法满足这种需求。
  • 会增加所有查找操作的时间,可能会导致测试运行时间增加。

代码示例:

import org.openqa.selenium.By;
import org.openqa.selenium.WebDriver;
import org.openqa.selenium.chrome.ChromeDriver;
import java.util.concurrent.TimeUnit;public class ImplicitWaitExample {public static void main(String[] args) {WebDriver driver = new ChromeDriver();// 设置隐式等待时间为10秒driver.manage().timeouts().implicitlyWait(10, TimeUnit.SECONDS);driver.get("http://xxxx.com");// 尝试查找元素,最多等待10s,找到就执行下面的,找不到就报错driver.findElement(By.cssSelector("xxx")).click(); // driver.quit();}
}

注意事项:

  1. 隐式等待会影响所有的findElementfindElements方法。
  2. 如果在查找元素时,设置的隐式等待时间较长,可能会影响测试执行效率。
更详细的例子:
示例说明

假设我们需要在一个页面上依次查找几个元素,并且都能在隐式等待时间内找到。这几个查找操作将依次进行,且每个操作都会应用隐式等待。

import org.openqa.selenium.By;
import org.openqa.selenium.WebDriver;
import org.openqa.selenium.WebElement;
import org.openqa.selenium.chrome.ChromeDriver;
import java.util.concurrent.TimeUnit;public class ImplicitWaitSequentialExample {public static void main(String[] args) {WebDriver driver = new ChromeDriver();// 设置隐式等待时间为10秒driver.manage().timeouts().implicitlyWait(10, TimeUnit.SECONDS);driver.get("http://xxx.com");// 第一次查找元素WebElement element1 = driver.findElement(By.cssSelector("#myElement1"));element1.click(); // 执行操作// 第二次查找元素WebElement element2 = driver.findElement(By.cssSelector("#myElement2"));element2.sendKeys("Hello"); // 执行操作// 修改隐式等待时间为5秒,后续的查找就是等待5sdriver.manage().timeouts().implicitlyWait(5, TimeUnit.SECONDS);// 第三次查找元素WebElement element3 = driver.findElement(By.cssSelector("#myElement3"));element3.sendKeys("Hello"); // 执行操作driver.quit();}
}
详细步骤
  1. 设置隐式等待

    driver.manage().timeouts().implicitlyWait(10, TimeUnit.SECONDS);
    

    这行代码设置了全局的隐式等待时间为10秒。

  2. 第一次查找元素

    WebElement element1 = driver.findElement(By.cssSelector("#myElement1"));
    

    WebDriver开始查找idmyElement1的元素。如果元素不存在,WebDriver会在10秒内反复尝试查找,直到找到元素或超时。

  3. 执行操作

    element1.click();
    

    一旦找到myElement1,立即执行点击操作。

  4. 第二次查找元素

    WebElement element2 = driver.findElement(By.cssSelector("#myElement2"));
    

    WebDriver开始查找idmyElement2的元素,同样会在10秒内反复尝试查找,直到找到元素或超时。

  5. 执行操作

    element2.sendKeys("Hello");
    

    一旦找到myElement2,立即执行发送文本操作。

  6. 修改等待时间

    driver.manage().timeouts().implicitlyWait(5, TimeUnit.SECONDS);
    

    修改后续的等待时间为5s。

  7. 执行操作

    WebElement element3 = driver.findElement(By.cssSelector("#myElement3"));
    

    WebDriver开始查找idmyElement2的元素,这里就是 5 秒内反复尝试查找,直到找到元素或超时。

总结

在设置了隐式等待后,WebDriver会在每次查找元素时,依次应用隐式等待。每个查找操作都是独立的,当前查找操作完成后才会进行下一个查找操作。因此,对于多个查找操作,它们会一个执行完再执行另一个,确保每个操作都能在指定的隐式等待时间内找到相应的元素并执行后续操作。


显式等待(Explicit Wait)

概述:

显式等待是在指定条件满足之前,WebDriver会等待特定的时间。常用的条件包括元素是否可见、是否可点击等。显式等待是针对具体的操作单独设置的,更加灵活。

优点:

  • 灵活,可以针对每个操作设置不同的等待条件和时间。
  • 提高测试的稳定性,只有在条件满足时才继续执行后续操作。

缺点:

  • 需要为每个操作单独编写等待逻辑,代码较为冗长。

代码示例:

import org.openqa.selenium.By;
import org.openqa.selenium.WebDriver;
import org.openqa.selenium.WebElement;
import org.openqa.selenium.chrome.ChromeDriver;
import org.openqa.selenium.support.ui.ExpectedConditions;
import org.openqa.selenium.support.ui.WebDriverWait;public class ExplicitWaitExample {public static void main(String[] args) {System.setProperty("webdriver.chrome.driver", "/path/to/chromedriver");WebDriver driver = new ChromeDriver();driver.get("http://example.com");// 设置显式等待,最多等待10秒钟WebDriverWait wait = new WebDriverWait(driver, 10);//级别是 s// 针对特定的条件进行等待,直到找到或超时WebElement element = wait.until(ExpectedConditions.presenceOfElementLocated(By.id("myElement")));// 继续执行后续操作element.click();driver.quit();}
}

wait.until()方法会先执行条件,如果条件成立(即元素变为可点击状态),则继续执行后续的代码;如果条件不成立,则会一直等待,直到超过预期的时间(即超时)为止。如果超时仍然没有找到符合条件的元素,则会抛出一个TimeoutException异常

常用等待条件:

  • presenceOfElementLocated(By locator): 等待元素出现,但不一定可见。
  • visibilityOfElementLocated(By locator): 等待元素可见。
  • elementToBeClickable(By locator): 等待元素可点击。

选择使用哪种等待

强制等待

  • 适用于调试阶段或极少数情况下,生产环境中不推荐使用。

隐式等待

  • 适用于整个应用程序的加载时间相对一致,且不需要针对具体操作单独设置等待时间的情况。
  • 一次设置,对整个WebDriver会话中的所有查找操作都生效。

显式等待

  • 适用于复杂场景,不同页面或操作需要不同的等待条件和时间。
  • 针对每个操作单独设置等待条件,更加灵活和精确。

总结

等待机制在自动化测试中至关重要,它们可以帮助确保测试脚本在执行操作前,页面元素已经加载完成。选择合适的等待机制,可以提高测试的可靠性和效率。

  • 强制等待:简单直接,但不推荐在生产环境中使用。
  • 隐式等待:适用于简单的场景,例如整个应用程序的加载时间相对一致,且不需要针对具体操作单独设置等待时间。
  • 显式等待:适用于复杂的场景,例如不同页面或操作需要不同的等待条件和时间。

通常,显式等待和隐式等待可以结合使用,但需要小心处理,因为它们会同时影响等待时间。例如,设置隐式等待10秒,再设置显式等待5秒,那么实际等待时间可能会超过预期。因此,在使用显式等待时,最好将隐式等待设置为较短时间或不设置隐式等待。

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

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

相关文章

【C++课程学习】:C++入门(输入输出,缺省参数)

🎁个人主页:我们的五年 🔍系列专栏:C课程学习 🎉欢迎大家点赞👍评论📝收藏⭐文章 目录 🍩1.关于C输入输出: 🍩2.缺省参数函数: 缺省参数的概…

聊聊大模型微调训练全流程的思考

前言 参考现有的中文医疗模型:MedicalGPT、CareGPT等领域模型的训练流程,结合ChatGPT的训练流程,总结如下: 在预训练阶段,模型会从大量无标注文本数据集中学习领域/通用知识;其次使用{有监督微调}(SFT)优化…

python如何快速的判断一个key在json的第几层呢,并修改其value值

python如何快速的判断一个key在json的第几层呢,并修改其value值 def find_and_modify_key(json_obj, target_key, new_value, current_level1):# 检查当前层是否包含目标keyif target_key in json_obj:print(f"找到 {target_key} 在第 {current_level} 层。&q…

在 JavaScript 中,除了使用可选链操作符,还有哪些方法可以安全地访问对象的深层属性?

在JavaScript中,除了使用可选链操作符(?.),还有几种方法可以安全地访问对象的深层属性,避免因访问不存在的属性而抛出错误。以下是一些常用的方法: 1. 条件(三元)运算符&#xff1…

以太网基础 -- LLDP使用案例

LLDP使用技术报告 背景 链路层发现协议(Link Layer Discovery Protocol,LLDP)是一种网络协议,主要用于在以太网网络中发现相邻设备并与其交换信息。LLDP是一种开放的标准,由IEEE 802.1AB定义,能够跨多个厂…

dubbo服务引用

spring刷新容器的时候,ReferrenceBean只有被调用的情况下才会在createBean回调afterPropertiesSet的时候引入服务。 ReferrenceBean实现了FacotryBean的接口,当对任意服务Interface进行自动注入或者getBean获取,会触发getObject过程 本地服…

TensorBoard在pytorch训练过程中如何使用,及数据读取问题解决方法

TensorBoard 模块导入日志记录文件的创建训练中如何写入数据如何提取保存的数据调用TensorBoard面板可能会遇到的问题 模块导入 首先从torch中导入tensorboard的SummaryWriter日志记录模块 from torch.utils.tensorboard import SummaryWriter然后导入要用到的os库&#xff0…

通过一个例子,说明Python的责任链设计模式有什么优缺点

责任链设计模式(Chain of Responsibility Pattern)是一种行为设计模式,它允许将一个请求沿着处理者链进行传递,直到有一个处理者处理它为止。在Python中,这种模式可以通过多种方式实现,通常涉及到一系列对象…

方案设计|汽车轮胎数显胎压计方案

一、引言 数显轮胎胎压计是一个专门测量车辆轮胎气压的工具,它具有高精度测量的功能,能够帮助快速准确获取轮胎气压正确数值,保证轮胎使用安全。本文将对数显轮胎胎压计的方案技术进行分析,包括其基本原理、硬件构成、软件设计等方…

架构学习:什么是业务架构图?如何画业务架构图?

6.1~6.18,艾威618年中大促,钜惠来袭!想报课但还没下手的小伙伴,都可以行动起来啦!活动规则还是一如既往的简单、粗暴——直接立减、返现、抽奖以及送礼品!了解活动详情,请点击这里》》 业务架构…

ActiViz中的vtkVolumeMapper

文章目录 一、简介二、工作原理三、核心功能四、使用步骤五、配置与优化六、案例与应用七、扩展与集成八、结论一、简介 vtkVolumeMapper是 ActiViz库中专为处理和渲染三维体积数据设计的核心组件。在诸如医学成像、地质科学、气象模型及工程仿真等多个领域,它扮演着至关重要…

【源码】SpringBoot事务注册原理

前言 对于数据库的操作,可能存在脏读、不可重复读、幻读等问题,从而引入了事务的概念。 事务 1.1 事务的定义 事务是指在数据库管理系统中,一系列紧密相关的操作序列,这些操作作为一个单一的工作单元执行。事务的特点是要么全…

实验9 浮动静态路由配置

--名称-- 一、 原理描述二、 实验目的三、 实验内容四、 实验配置五、 实验步骤 一、 原理描述 浮动静态路由也是一种特殊的静态路由,主要考虑链路冗余。浮动静态路由通过配置一条比主路由优先级低的静态路由,用于保证在主路由失效的情况下,…

代码随想录 day27|day28|day29

回溯2 切割问题:是在每个节点判断是否是要剪枝收割元素。 startidx 是切割起点,i是本次切割终点 分割回文串 复原ip地址 非递减子序列 都是在树的节点依照题意判断,之后决定是否剪枝。 也就是都有if判断来剪枝 。 下面是非递减子序列。 下…

Python | 虚拟环境的增删改查

mkvirtualenv创建虚拟环境 mkvirtualenv是用于在Pyhon中创建虚拟环境的命令。它通过使用vitualenv库来创建一个隔离的Python环境,以便您可以安装特定版本的Python包,而不会影响全局Python环境。 使用方法: 安装virtualenv:pip install vir…

前端将xlsx转成json

第一种方式,用js方式 1.1先安装插件 万事都离不开插件的支持首先要安装两个插件 1.2. 安装xlsx cnpm install xlsx --save注:这块我用的cnpm,原生的是npm,因为镜像的问题安装了cnpm,至于怎么装网上一搜一大堆 1.3安…

用langchain搭配最新模型ollama打造属于自己的gpt

langchain 前段时间去玩了一下langchain,熟悉了一下大模型的基本概念,使用等。前段时间meta的ollama模型发布了3.0,感觉还是比较强大的,在了解过后,自己去用前后端代码,调用ollama模型搭建了一个本地的gpt应用。 核心逻辑 开始搭…

Vue 封装elementUI的el-popover

1.封装公共组件 <template><div class"confirm-popover disInlineBlock ml10"><el-popover placement"bottom" v-model"visible" :trigger"triggerType"><div class"confirm-popover-popover"><…

vue3中进度条上加高亮圆点

实现效果 小圆点基于进度条定位&#xff08;left&#xff09;。 实现代码 <template><!-- 这块代码实现的功能&#xff1a;progressData遍历的年份进度数组&#xff0c;展示每年完成的进度--><ul><li v-for"(item, index) in progressData" :k…

Unity VR 零基础开发之 Pico4 MR

一、新建Unity2021.3.37 3D工程 二、切换到Android安卓平台 1、点击Unity编辑器左上角的Flie后&#xff0c;选择Build Setting选项。 2、弹出弹窗后&#xff0c;点击Android选项&#xff0c;然后再点击Switch Platform按钮切换成安卓平台。 3、切换完成后Android选项后面会显示…