前言:因为浏览器元素渲染速度和脚本运行速度是不一致的,时常会发生脚本运行报错提示元素找不到的情况,selenium提供了三种等待方式增加等待时长以避免以上情况,分别是强制等待、隐式等待、显式等待。
一、强制等待
使用time模块的sleep(),强制代码等待设定的时间,时间到了再执行下面的代码。这是最简单粗暴的一种方法,但是影响代码运行速度,并且在无法精准地判断元素加载所需时长时会使代码运行报错,如设置时间小于加载时间。
from selenium import webdriver
from time import sleep
driver = webdriver.Chrome()
driver.get("//www.greatytc.com/u/a2d1bc176e90")
sleep(3)
二、隐式等待
使用implicitly_wait()方法,设置一个等待时间,如果等待时间内页面加载完成则进行下一步。不过隐式等待不好的一点是需要等待整个页面加载完成再进行下一步,而不是自己设定的元素加载完成就进行下一步操作。
隐式等待的作用是全局的,仅需设置一次。
from selenium import webdriver
driver = webdriver.Chrome()
driver.implicitly_wait(5)
driver.get("//www.greatytc.com/u/a2d1bc176e90")
三、显式等待
使用WebDriverWait()方法,针对于特定元素设置等待时间,在设置时间内,默认每隔一段时间检测一次当前页面某个元素是否存在,如果在规定的时间内找到了元素,就执行相关操作,如果超过设置时间检测不到则抛出异常。
WebDriverWait()一般由until()或until_not方法配合使用,下面是这两种方法的说明:
until(method,message=''): 调用该方法提供的驱动程序作为一个参数,直到返回值为True;
until_not(method,message=''): 调用该方法提供的驱动程序作为一个参数,直到返回值为Flase;
格式如下:
WebDriverWait(driver,timeout,poll_frequency=0.5,ignored_exceptions=None)
driver:浏览器驱动
timeout:最长超时时间
poll_frequency:检测间隔时间,默认0.5s
ignored_exceptions:超时后的异常信息,默认情况抛出NoSuchElementException异常
示例代码如下:
from selenium import webdriver
from selenium.webdriver.support.wait import WebDriverWait
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.common.by import By
driver = webdriver.Chrome()
driver.implicitly_wait(10) #隐性等待和显性等待可以同时用,但要注意:等待的最长时间取两者之中的大者
driver.get("//www.greatytc.com/u/a2d1bc176e90")
ele = (By.LINK_TEXT, "Selenium")
try:
WebDriverWait(driver, 20, 0.5).until(EC.presence_of_element_located(ele))
print(driver.find_element_by_link_text("Selenium").get_attribute("href"))
finally:
driver.close()
expected_conditions类提供的预期条件判断方法如下:
title_is: 判断当前页面的title是否完全等于(==)预期字符串,返回布尔值
title_contains : 判断当前页面的title是否包含预期字符串,返回布尔值
presence_of_element_located : 判断某个元素是否被加到了dom树里,并不代表该元素一定可见
visibility_of_element_located : 判断某个元素是否可见. 可见代表元素非隐藏,并且元素的宽和高都不等于0
visibility_of : 跟上面的方法做一样的事情,只是上面的方法要传入locator,这个方法直接传定位到的element就好了
presence_of_all_elements_located : 判断是否至少有1个元素存在于dom树中。举个例子,如果页面上有n个元素的class都是‘column-md-3‘,那么只要有1个元素存在,这个方法就返回True
text_to_be_present_in_element : 判断某个元素中的text是否 包含 了预期的字符串
text_to_be_present_in_element_value : 判断某个元素中的value属性是否 包含 了预期的字符串
frame_to_be_available_and_switch_to_it : 判断该frame是否可以switch进去,如果可以的话,返回True并且switch进去,否则返回False
invisibility_of_element_located : 判断某个元素中是否不存在于dom树或不可见
element_to_be_clickable : 判断某个元素中是否可见并且是enable的,这样的话才叫clickable
staleness_of : 等某个元素从dom树中移除,注意,这个方法也是返回True或False
element_to_be_selected : 判断某个元素是否被选中了,一般用在下拉列表
element_selection_state_to_be : 判断某个元素的选中状态是否符合预期
element_located_selection_state_to_be : 跟上面的方法作用一样,只是上面的方法传入定位到的element,而这个方法传入locator
alert_is_present : 判断页面上是否存在alert
需要注意的一点是,一些EC(预期条件)方法使用locators,而一些使用elements。如上面实例代码中的presence_of_element_located()是使用locators,所以入参的时候不能传入elements,否则会报错。