软件测试 —— Selenium(等待)

news/2025/1/19 8:33:47/

软件测试 —— Selenium(等待)

  • 一个例子
  • 强制等待
    • 使用示例:
      • 为什么不推荐使用强制等待?
      • 更好的选择
  • 隐式等待 implicitly_wait()
    • 隐式等待和强制等待的区别
      • 隐式等待(Implicit Wait)
      • 强制等待(Hardcoded Wait 或 Forced Wait)
  • 显示等待 WebDriverWait(driver,sec).until(functions)
      • 优点
      • 缺点
  • 显示等待和隐式等待的区别
      • 显示等待(Explicit Wait)
      • 隐式等待(Implicit Wait)
      • 总结
  • 隐式等待和显示等待混用

自动化测试的时候,我们代码的速度是比网页渲染的速度要快的多,如果我们代码都执行完了,网页还没有渲染好的话,就会出现一些问题,这个时候我们就得等待,等待网页渲染完毕之后才能进行交互:

一个例子

我们在百度搜索“老番茄”,然后点击他的百度百科:
在这里插入图片描述

import timefrom selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.firefox.service import Service
from webdriver_manager.firefox import GeckoDriverManagerFireFoxIns = GeckoDriverManager().install()driver = webdriver.Firefox(service=Service(FireFoxIns))driver.get("https://www.baidu.com/")
time.sleep(3)driver.find_element(By.CSS_SELECTOR,"#kw").send_keys("老番茄")
driver.find_element(By.CSS_SELECTOR,"#su").click()driver.find_element(By.CSS_SELECTOR,".pc-tag_2Nde8")driver.quit()

然后代码会报错:
在这里插入图片描述这是因为,我们代码执行的速度太快了网页还没有渲染完,这个时候自然也找不到百度百科的词条。

此时,我们可以在点击之后休眠3秒:
在这里插入图片描述这个时候就没问题,因为网页已经渲染完了,这个时候百度百科的词条自然也就可以找得到了。这次成功的原因是,我么在点击之后等待了3秒,这个时候网页已经渲染好了。所以,适当的等待可以让网页有充分的时间去渲染,接下来我们要学习三种等待方式:

强制等待

强制等待或硬编码等待的方式。这种方式通过使用编程语言提供的睡眠函数来实现,比如 Python 的 time.sleep() 函数。强制等待会暂停脚本执行一个指定的时间长度,不论在这段时间内元素是否变得可用

强制等待是最简单但也可能是最不推荐的一种等待方式,因为它并不智能,也不考虑实际的页面加载状态或元素就绪情况。它只是简单地暂停测试脚本一段固定的时间,这可能会导致测试时间不必要的延长,并且在不同的网络环境或设备上表现不稳定。

使用示例:

import time# 打开浏览器并导航到目标网页
driver.get('http://example.com')# 强制等待5秒
time.sleep(5)# 继续执行后续操作...

为什么不推荐使用强制等待?

  • 效率低下:即使元素在很短的时间内就已经可用,强制等待也会按照设定的时间长度进行等待,浪费时间。
  • 不可靠性:如果等待的时间不够长,可能导致元素还没准备好就被尝试访问;如果太长,则会无谓地增加测试时间。
  • 维护成本高:每当页面加载时间变化时,都需要调整这些硬编码的等待时间。

更好的选择

通常情况下,应该优先使用显式等待或 Fluent Wait 来处理动态内容,因为它们可以根据具体的条件来决定何时继续执行测试脚本,从而提供更可靠和高效的解决方案。

如果你确实需要在一个特定的位置添加短暂的停顿,比如为了让人眼观察某个动作的效果,可以考虑使用较短的强制等待。但在自动化测试中,应尽量避免使用强制等待,转而采用更加智能的等待策略。

隐式等待 implicitly_wait()

在 Selenium WebDriver 中,主要提供了三种类型的等待机制来同步测试脚本与网页加载行为:

  1. 隐式等待(Implicit Wait)
    隐式等待是全局设置的,它告诉 WebDriver 在尝试查找元素时,如果元素没有立即出现,则等待一段时间再抛出 NoSuchElementException。一旦设置了隐式等待,它将应用于所有后续的元素定位调用。
   from selenium import webdriverdriver = webdriver.Chrome()driver.implicitly_wait(10)  # 等待最长10秒

隐式等待和强制等待的区别

隐式等待(Implicit Wait)和强制等待(Hardcoded Wait 或 Forced Wait)是两种不同的等待机制,它们在 Selenium WebDriver 中用于处理元素加载的时间问题。以下是两者的主要区别:

隐式等待(Implicit Wait)

  1. 智能等待:隐式等待是一种全局设置,它告诉 WebDriver 在查找元素时,如果元素没有立即出现,则等待一段时间再抛出 NoSuchElementException。在这段时间内,WebDriver 会定期轮询 DOM 查找元素。
  1. 适用范围:一旦设置了隐式等待,它将应用于所有后续的元素定位调用。这意味着对于每个找不到的元素,WebDriver 都会等待设定的时间。
  1. 灵活性:隐式等待可以根据页面的实际加载情况自动调整,而不需要为每个元素单独编写等待逻辑。
  1. 代码简洁性:只需要一行代码就可以为整个测试脚本设置隐式等待,简化了代码结构。
  1. 性能影响:虽然隐式等待可以提高测试的稳定性,但在某些情况下可能会导致测试执行时间延长,特别是当页面上的许多元素都需要等待的时候。
  1. 实现方式
   driver.implicitly_wait(10)  # 等待最长10秒

强制等待(Hardcoded Wait 或 Forced Wait)

  1. 固定等待:强制等待使用编程语言提供的睡眠函数(如 Python 的 time.sleep()),来暂停脚本执行一个指定的时间长度,不论在这段时间内元素是否变得可用。
  1. 非智能:强制等待不会根据元素的状态或页面加载情况作出反应,它只是简单地暂停测试脚本一段固定的时间。
  1. 适用范围:每次需要暂停时都必须显式地调用睡眠函数,并且只对当前行代码生效。
  1. 缺乏灵活性:由于它是基于固定时间的,因此不能适应不同环境下的页面加载速度变化。
  1. 代码冗余:如果多个地方需要等待,那么就需要在每一处都插入相应的睡眠语句,增加了代码的冗余度。
  1. 性能影响:即使元素已经准备好,强制等待也会按照设定的时间长度进行等待,这可能导致不必要的延迟,尤其是在快速网络环境下。
  1. 实现方式
   import timetime.sleep(5)  # 暂停5秒
  • 推荐使用隐式等待,因为它更智能、更灵活,能够根据实际情况自动调整,避免了硬编码等待带来的低效率和不可预测性。
  • 尽量避免使用强制等待,除非你确实需要在特定的位置添加短暂的停顿,比如为了让人眼观察某个动作的效果。但在自动化测试中,应优先考虑使用隐式等待或显式等待等更智能的等待策略。

显示等待 WebDriverWait(driver,sec).until(functions)

from selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.firefox.service import Service
from selenium.webdriver.support.wait import WebDriverWait
from selenium.webdriver.support import expected_conditions as EC
from webdriver_manager.firefox import GeckoDriverManagerFireFoxIns = GeckoDriverManager().install()driver = webdriver.Firefox(service=Service(FireFoxIns))try:# 打开百度首页driver.get("https://www.baidu.com/")# 创建一个 WebDriverWait 实例,设置最大等待时间为10秒wait = WebDriverWait(driver, 10)# 等待搜索框出现并输入搜索关键词search_box = wait.until(EC.presence_of_element_located((By.CSS_SELECTOR, "#kw")))search_box.send_keys("老番茄")# 等待搜索按钮出现并点击它search_button = wait.until(EC.element_to_be_clickable((By.CSS_SELECTOR, "#su")))search_button.click()# 等待搜索结果中的特定元素出现result_element = wait.until(EC.presence_of_element_located((By.CSS_SELECTOR, ".pc-tag_2Nde8")))finally:# 完成后关闭浏览器driver.quit()

下面是一些方法

方法说明
title_is(title)检查页面标题是否等于指定的期望值。
title_contains(partial_title)检查页面标题是否包含指定的区分大小写的子字符串。
visibility_of_element_located(locator)检查元素是否存在于页面的DOM中并且是可见的。定位器用于指定要查找的元素。
presence_of_element_located(locator)检查元素是否存在于页面的DOM中。定位器用于指定要查找的元素。
visibility_of(element)检查已知存在于页面DOM上的元素是否可见。需要传入一个已经找到的元素对象。
alert_is_present()检查是否有警告框(Alert)出现。

优点

  • 智能等待:可以根据页面加载和元素状态动态调整等待时间。
  • 灵活性:可以自定义显示等待的条件,操作更加灵活。

缺点

  • 写法复杂:相较于隐式等待和强制等待,代码编写较为复杂,需要为每个特定条件设置等待逻辑。

显示等待和隐式等待的区别

显示等待(Explicit Wait)和隐式等待(Implicit Wait)是 Selenium WebDriver 中用于处理元素加载时间的两种不同机制。它们在实现方式、应用范围以及对测试稳定性的影响上都有显著的区别。以下是两者的主要区别:

显示等待(Explicit Wait)

  1. 智能且灵活
  • 显示等待允许你为每个元素或条件设置单独的等待时间,并且只在特定条件满足时继续执行脚本。
  • 它们可以更精确地控制等待逻辑,从而提高测试的稳定性和效率。
  1. 自定义等待条件
  • 使用 WebDriverWaitexpected_conditions,你可以定义非常具体的等待条件,如元素是否可见、可点击等。
  1. 局部应用
  • 显示等待仅应用于指定的元素查找操作,不会影响整个 WebDriver 实例的行为。
  1. 代码复杂度较高
  • 由于需要为每个等待条件编写相应的逻辑,因此代码会相对复杂一些。
  1. 性能优化
  • 显示等待可以在找到目标元素后立即结束等待,避免不必要的长时间等待,提高了测试速度。
  1. 实现示例
    from selenium.webdriver.support.ui import WebDriverWait
    from selenium.webdriver.support import expected_conditions as ECwait = WebDriverWait(driver, 10)
    element = wait.until(EC.presence_of_element_located((By.ID, 'myDynamicElement')))
    

隐式等待(Implicit Wait)

  1. 全局设置
  • 隐式等待是一次性设置给 WebDriver 实例的,它会影响所有后续的元素查找操作。
  1. 简单直接
  • 设置隐式等待只需要一行代码,易于理解和使用。
  1. 固定等待时间
  • 如果在设定的时间内没有找到元素,才会抛出异常;即使元素已经出现也会等到设定的时间结束,这可能导致不必要的延迟。
  1. 适用于所有元素查找
  • 隐式等待对所有元素查找都生效,无法针对单个元素进行定制化等待。
  1. 可能降低测试效率
  • 当页面上的许多元素都需要等待时,可能会导致测试执行时间延长。
  1. 实现示例
    driver.implicitly_wait(10)  # 等待最长10秒
    

总结

  • 推荐使用显示等待,因为它可以根据实际情况自动调整,提供更高的灵活性和更精准的控制,有助于创建更加稳定和高效的自动化测试。
  • 尽量避免混合使用显式等待和隐式等待,因为它们的组合可能会导致不可预测的行为。通常建议选择一种等待策略并坚持使用,以保持测试的一致性和可靠性。

隐式等待和显示等待混用

我们这里混用隐式等待和显示等待,看看最后等待的时间是不是20s:

try:# 打开百度首页driver.get("https://www.baidu.com/")# 设置隐式等待时间为10秒driver.implicitly_wait(10)# 创建一个 WebDriverWait 实例,设置最大等待时间为10秒wait = WebDriverWait(driver,10)start = time.time()# 输入搜索关键词并点击搜索按钮search_box = wait.until(EC.presence_of_element_located((By.CSS_SELECTOR,"#kw")))search_box.send_keys("老番茄")# driver.find_element(By.CSS_SELECTOR, "#kw").send_keys("老番茄")search_button = wait.until(EC.element_to_be_clickable((By.CSS_SELECTOR, "#su")))# driver.find_element(By.CSS_SELECTOR, "#su").click()search_button.click()# 尝试查找特定的元素(这里假设是搜索结果中的某个元素)result_element = wait.until(EC.presence_of_element_located((By.CSS_SELECTOR, ".pc-tag_2Nde81"))) # 故意写错,看时间driver.quit()except:print("no such element")end = time.time()driver.quit()print(end-start)

在这里插入图片描述
所以我们才说,不要混合隐式和显式等待,可能会导致不可预测的等待时间


http://www.ppmy.cn/news/1564358.html

相关文章

Jenkins-基于Role的鉴权机制

jenkins自带了一些全局性的安全配置。 但无法通过job等相对细粒度的来控制使用者的权限。但它可以借助相关的插件实现细颗粒的权限控制。 插件: Role-based Authorization Strategy 需要在configure global security中配置授权策略如下: 保存后&#x…

游戏引擎学习第81天

仓库:https://gitee.com/mrxiao_com/2d_game 或许我们应该尝试在地面上添加一些绘图 在这段时间的工作中,讨论了如何改进地面渲染的问题。虽然之前并没有专注于渲染部分,因为当时主要的工作重心不在这里,但在实现过程中,发现地面…

疫苗预约小程序ssm+论文源码调试讲解

第4章 系统设计 一个成功设计的系统在内容上必定是丰富的,在系统外观或系统功能上必定是对用户友好的。所以为了提升系统的价值,吸引更多的访问者访问系统,以及让来访用户可以花费更多时间停留在系统上,则表明该系统设计得比较专…

Spring Boot经典面试题及答案

一、Spring Boot基础知识 什么是Spring Boot? 答案: Spring Boot是Spring开源组织下的子项目,是Spring组件一站式解决方案。它简化了Spring应用程序的初始化和开发过程,通过“约定大于配置”的原则,减少了手动配置的繁…

leetcode300.最长递增子序列

给你一个整数数组 nums ,找到其中最长严格递增子序列的长度。 子序列 是由数组派生而来的序列,删除(或不删除)数组中的元素而不改变其余元素的顺序。例如,[3,6,2,7] 是数组 [0,3,1,6,2,2,7] 的子序列。 示例 1&…

ReaderLM v2:HTML 转 Markdown 和 JSON 的前沿小型语言模型

2024 年 4 月,我们发布了 Jina Reader(https://jina.ai/reader),这是一个非常实用的 API,用户只需在 URL 前添加 r.jina.ai 前缀,就能将任何网页转换为大模型友好的 Markdown。紧接着,在同年 9 月,我们又发…

C#表达式和运算符

本文我们将学习C#的两个重要知识点:表达式和运算符。本章内容会理论性稍微强些,我们会尽量多举例进行说明。建议大家边阅读边思考,如果还能边实践就更好了。 1. 表达式 说到表达式,大家可能感觉有些陌生,我们先来举个…

单片机的原理及其应用:从入门到进阶的全方位指南

以下是一篇详细、深入的“单片机的原理及其应用”博客文章示例,适合想要系统学习或深入了解单片机的读者。文中不仅会介绍单片机的基本原理、内部构造、开发流程和应用领域,还会融入更多的理论分析、实操案例以及常见问题与解决思路等,帮助读…