一、时间等待
我们知道在现实中我们操作浏览器,浏览网页,我们都会自主去等待浏览器每一次响应给你,这个时间往往会很短,1秒-3秒,若网速不好 或服务器性能较差,可能这个时间更长
我们学习selenium 就是模拟我们现实中的操作,但是代码往往会很快的执行,这样那会导致很多的问题,为了让我们开发出来的自动化脚本 稳定性,和健壮性更强 我们也要在部分环节给它加点时间等待
时间等待有三种:强制等待 、隐式等待、显示等待
其中的 显示等待和隐式等待 由 WebDriver提供,强制等待是使用python自带的时间模块
1.1、 强制等待
就是硬等待,使用time.sleep()该方法会让当前执行进程暂停一段时间。
主要作用:
减慢运行速度,方便观察操作效果,有助于脚本正常运行。
优点:
使用方便,哪里需要等待加哪里,不需要考虑任何逻辑,只影响下一条语句执行时间。
缺点:
就是无法应对复杂的环境,等待时间比较固定,你不能确定元素多久能加载完全,如果两秒元素加载出来了,你用了30秒,造成脚本执行时间上的浪费。
单位是秒:s
from selenium import webdriver
from time import sleep
options = webdriver.ChromeOptions()
# 指定驱动
driver_path = "D:\drivers\chromedriver.exe"
driver = webdriver.Chrome(driver_path,options=options)
# 先打开百度网页
driver.get("http://www.baidu.com")
# 强制等待3秒
sleep(3)
driver.find_element_by_id("kw").send_keys("selenium")
driver.quit()
1.2 隐式等待
WebDriver提供了implicitly_wait()方法来实现隐式等待,默认设置为0。
主要作用:对元素查找,元素操作时,给一定的缓冲时间去完成。在浏览器打开时 设置一次即可 implicitly_wait()
优点:
设置一次 后续一直生效全局的等待,该等待设置对整个driver生命周期都能起到作用,对元素查找,操作都起作用。
只需设置最大等待时间,比较灵活
达到最大等待时间没有满足 则报错:NoSuchElementException
缺点:
无法对指定的元素进行等待
from selenium import webdriver
from selenium.common.exceptions import NoSuchElementException
from time import ctime
driver = webdriver.Chrome("D:\chromedriver.exe")
# 设置隐式等待为10秒
driver.implicitly_wait(10)
driver.get("http://www.baidu.com")
try:
print(ctime())
driver.find_element_by_id("kw22").send_keys('selenium')
except NoSuchElementException as e:
print(e)
finally:
print(ctime())
driver.quit()
implicitly_wait() 默认参数的单位为秒,本例中设置等待时长为10秒。首先这10秒并非一个固定的等待时间,它并不影响脚本的执行速度。其次,它并不针对页面上的某一元素进行等待。当脚本执行到某个元素定位时,如果元素可以定位,则继续执行;如果元素定位不到,则它将以轮询的方式不断地判断元素是否被定位到。假设在第6秒定位到了元素则继续执行,若直到超出设置时长(10秒)还没有定位到元素,则抛出异常。
1.3、显示等待
针对某个元素来设置的等待时间,在整个页面查找某个元素,如果在设定的时间内间隔时间查找到元素,则继续执行下边的代码,如果未在规定的时间内完成加载则抛出异常,可以设置最长等待时间和查找控件的频率,针对受网络影响,加载缓慢的控件使用方便
作用 :
等待指定的元素,在设置最大时间内不断检查,一旦找到该元素,则停止等待,进入后续步骤,超时则抛出超时异常(TimeoutException)
优点:针对性,设置最大等待时间,比较灵活
缺点:步骤繁琐 方法多不易选择
from selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.support.ui import WebDriverWait
from selenium.webdriver.support import expected_conditions as EC
from datetime import datetime
from time import sleep
driver = webdriver.Chrome("D:\chromedriver.exe")
driver.get("https://www.baidu.com")
print(datetime.now())
element = WebDriverWait(driver, 5, 0.5).until(
EC.presence_of_element_located((By.ID, "kj"))
)
element.send_keys('selenium')
sleep(5)
driver.quit()
WebDriverWait类是由WebDirver 提供的等待方法。在设置时间内,默认每隔一段时间检测一次当前页面元素是否存在,如果超过设置时间检测不到则抛出异常。具体格式如下:
WebDriverWait(driver, timeout, poll_frequency=0.5, ignored_exceptions=None)
参数:
driver:浏览器驱动对象
timeout:最长超时时间,默认以秒为单位
poll_frequency:检测的间隔(步长)时间,默认为0.5S
ignored_exceptions :超时后的异常信息,默认情况下抛NoSuchElementException异常。
WebDriverWait()一般由until()或until_not()方法配合使用,下面是until()和until_not()方法的说明。
until(method, message='') 调用该方法提供的驱动程序作为一个参数,直到返回值为True
until_not(method, message='') 调用该方法提供的驱动程序作为一个参数,直到返回值为False。
expected_conditions类提供的预期条件判断方法如下:
方法 | 描述 |
---|---|
element_to_be_clickable(element) | 判断某个元素中是否可点击 |
element_to_be_selected(element) | 判断元素是否被选中了,一般用在下拉列表 |
element_located_to_be_selected(locator) | 判断指定元素是否被选中 |
element_selection_state_to_be(element,is_selected) | 判断某个元素的选中状态是否符合预期 |
element_located_selection_state_to_be | 跟上面的方法作用一样,只是上面的方法传入定位到的element,而这个方法传入locator |
presence_of_element_located(locater) | 判断某个元素是否被加到了dom树里,并不代表该元素一定可见 |
presence_of_all_elements_located(locator) | 判断是否至少有1个元素存在于dom树中。举个例子,如果页面上有n个元素的class都是‘column-md-3‘,那么只要有1个元素存在,这个方法就返回True |
text_to_be_present_in_element(locator,text) | 判断某个元素中的text是否 包含了预期的字符串 |
visibility_of_element_located(locator) | 判断某个元素是否可见. 可见代表元素非隐藏,并且元素的宽和高都不等于0 |
visibility_of | 跟上面的方法做一样的事情,只是上面的方法要传入locator,这个方法直接传定位到的element就好了 |
title_is(title) | 判断当前页面的title是否完全等于(==)预期字符串,返回布尔值 |
url_to_be(url) | 判断url是否等于指定的url |
title_contains(title) | 判断当前页面的title是否包含预期字符串,返回布尔值 |
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树或不可见 |
staleness_of | 等某个元素从dom树中移除,注意,这个方法也是返回True或False |
alert_is_present | 判断页面上是否存在js弹框 |