38、Selenium 之 WebDriverWait

之前有篇文章介绍过Selenium的等待知识,可惜当时水平有限,只介绍了sleep和implicitly_wait的用法。今天还是补充一下WebDriverWait的知识吧。

一、webdrivewait 示例代码

from selenium import webdriver
from selenium.webdriver.support.ui import WebDriverWait
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.common.by import By
import time

driver = webdriver.Chrome()
driver.get("http://www.baidu.com/")
locator = (By.ID,"kw")

try:
    ele = WebDriverWait(driver,10).until(EC.presence_of_element_located(locator))
    driver.find_element_by_id("kw").send_keys('abc')
    time.sleep(1) #为了看效果
except:
    print("ele can't find")
finally:
    driver.quit()
接下来一点点分析下这个语句
WebDriverWait(driver,10).until(EC.presence_of_element_located(locator))

(1)WebDriverWait 方法

driver: 传入WebDriver实例,即我们上例中的driver
timeout: 超时时间,等待的最长时间(同时要考虑隐性等待时间)
poll_frequency: 调用until或until_not中的方法的间隔时间,默认是0.5秒
ignored_exceptions: 忽略的异常,如果在调用until或until_not的过程中抛出这个元组中的异常,
        则不中断代码,继续等待,如果抛出的是这个元组外的异常,则中断代码,抛出异常。默认只有NoSuchElementException。

(2)WebDriverWait 方法后面有两种等待方式


until

method: 在等待期间,每隔一段时间(__init__中的poll_frequency)调用这个传入的方法,直到返回值不是False
message: 如果超时,抛出TimeoutException,将message传入异常
until_not

与until相反,until是当某元素出现或什么条件成立则继续执行,
 until_not是当某元素消失或什么条件不成立则继续执行,参数也相同,不再赘述。

看了以上内容基本上很清楚了,调用方法如下:

WebDriverWait(driver, 超时时长, 调用频率, 忽略异常).until(可执行方法, 超时时返回的信息)
 
  • 1

这里需要特别注意的是until或until_not中的可执行方法method参数,很多人传入了WebElement对象,如下:

WebDriverWait(driver, 10).until(driver.find_element_by_id('kw'))  # 错误
 
  • 1

这是错误的用法,这里的参数一定要是可以调用的,即这个对象一定有 __call__() 方法,否则会抛出异常:

TypeError: 'xxx' object is not callable
 
  • 1

在这里,你可以用selenium提供的 expected_conditions 模块中的各种条件,也可以用WebElement的 is_displayed() 、is_enabled()is_selected() 方法,或者用自己封装的方法都可以,那么接下来我们看一下selenium提供的条件有哪些:

expected_conditions



expected_conditions是selenium的一个模块,其中包含一系列可用于判断的条件:

selenium.webdriver.support.expected_conditions(模块)

 
  • 1
  • 2

以下两个条件类验证title,验证传入的参数title是否等于或包含于driver.title 
title_is 
title_contains

以下两个条件验证元素是否出现,传入的参数都是元组类型的locator,如(By.ID, ‘kw’) 
顾名思义,一个只要一个符合条件的元素加载出来就通过;另一个必须所有符合条件的元素都加载出来才行 
presence_of_element_located 
presence_of_all_elements_located

以下三个条件验证元素是否可见,前两个传入参数是元组类型的locator,第三个传入WebElement 
第一个和第三个其实质是一样的 
visibility_of_element_located 
invisibility_of_element_located 
visibility_of

以下两个条件判断某段文本是否出现在某元素中,一个判断元素的text,一个判断元素的value 
text_to_be_present_in_element 
text_to_be_present_in_element_value

以下条件判断frame是否可切入,可传入locator元组或者直接传入定位方式:id、name、index或WebElement 
frame_to_be_available_and_switch_to_it

以下条件判断是否有alert出现 
alert_is_present

以下条件判断元素是否可点击,传入locator 
element_to_be_clickable

以下四个条件判断元素是否被选中,第一个条件传入WebElement对象,第二个传入locator元组 
第三个传入WebElement对象以及状态,相等返回True,否则返回False 
第四个传入locator以及状态,相等返回True,否则返回False 
element_to_be_selected 
element_located_to_be_selected 
element_selection_state_to_be 
element_located_selection_state_to_be

最后一个条件判断一个元素是否仍在DOM中,传入WebElement对象,可以判断页面是否刷新了 
staleness_of

上面是所有17个condition,与until、until_not组合能够实现很多判断,如果能自己灵活封装,将会大大提高脚本的稳定性。

封装示例:

待补充


已标记关键词 清除标记
大家好! 我现在用Selenium的WebDriver的显式等待来判断元素是否出现,关于WebDriverWait对象定义的超时时间的计算方法我不太明白,请教一下: 如果Java的代码这样写: WebDriver webPage = new InternetExplorerDriver(); //验证网页title是否正确 try{ WebDriverWait waitTitle = new WebDriverWait(webPage,10); waitTitle.until(ExpectedConditions.titleIs("XX平台")); System.out.println("Title正确。"); } catch(TimeoutException t){ System.out.println("Title错误。"); } //检查banner是否可见 try{ WebDriverWait waitBanner = new WebDriverWait(webPage,10); waitBanner.until(ExpectedConditions.visibilityOf(webPage.findElement(By.cssSelector(".banner")))); System.out.println("banner可见。"); } catch(TimeoutException t){ System.out.println("banner不可见。"); } 这样写应该没问题,就是每检查一项内容,都要去new一个WebDriverWait对象。我刚开始学Java,还不太懂内存是怎么回收的。如果检查的内容较多,内存肯定会占用得越来越多。不知这些内存什么时候会被系统回收? 后来我想改成这样的写法: WebDriver webPage = new InternetExplorerDriver(); WebDriverWait wait = new WebDriverWait(webPage,10); //验证网页title是否正确 try{ wait.until(ExpectedConditions.titleIs("XX平台")); System.out.println("Title正确。"); } catch(TimeoutException t){ System.out.println("Title错误。"); } //检查banner是否可见 try{ wait.until(ExpectedConditions.visibilityOf(webPage.findElement(By.cssSelector(".banner")))); System.out.println("banner可见。"); } catch(TimeoutException t){ System.out.println("banner不可见。"); } 这样只实例化了一个WebDriverWait对象,内存倒是不会占用那么多了,但是我不太清楚超时时间是怎样计算的。在new WebDriverWait对象时,定义了超时时间是10秒,先判断网页title是否正确时,会占用一部分时间;那么,再判断banner是否可见,这个时间是重新从0开始计时,还是接着上一次消耗完了的时间继续计时呢?比如判断网页title耗掉了1秒,那么再判断banner是否可见,剩余可用的超时时间,是9秒,还是10秒呢?如果是接着上一次消耗的计时,即只剩9秒,就不是我的本意了,我想让每次判断都有10秒的超时时间,那是不是只能采用第一种写法?如果这样,内存需要控制吗?麻烦高人指教!谢谢!!!
©️2020 CSDN 皮肤主题: 大白 设计师:CSDN官方博客 返回首页