前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >python selenium2示例 - 同步机制

python selenium2示例 - 同步机制

作者头像
苦叶子
发布2018-04-04 16:03:20
8150
发布2018-04-04 16:03:20
举报
文章被收录于专栏:开源优测开源优测开源优测

前言

在使用python selenium2进行自动化测试实践的过程中,经常会遇到元素定位不到,弹出框定位不到等等各种定位不到的情况,在大多数的情况下,无非是以下两种情况:

1、有frame存在,定位前,未switch到对应的frame内 2、元素未加载完毕(从界面看已经显示),但DOM树还在load状态或在加载js

那对于这类情况,怎么解决呢?

通俗的讲法: 等待。

高大上点:解决自动化测试代码与浏览器加载渲染之间的同步问题。

下面我们分段讲述各种处理方式:

一 强制等待

这种方式简单粗暴直接有效,不足的地方就是不够灵活。下面看下代码片段:

#_*_ coding:utf-8 _*_ __author__ = '苦叶子' from selenium import webdriver from time import sleep # 注意 if __name__ == '__main__': driver = webdrivrer.Firefox() driver.get('http://www.testingunion.com') sleep(3) # 强制等待3s在执行下一步 print u"当前url: ", driver.current sleep(3) # 强制等待3s在执行下一步 driver.quit()

注: 请注意加粗有删除线的代码行,用于实现强制等待

二、隐性等待

webdriver提供了基础的同步方法,隐性等待implicitly_wait(xx),该方式的意义是:不论业务代码运行在那一步,都需要等待webdriver xx秒,如果在等待的xx秒内 webdriver完成了对应的动作,则业务代码和webdriver都正常继续执行;如果超过了xx秒,webdriver还未完成对应的动作,则业务代码继续执行,而webdriver则会抛出异常(例如timeout或元素未找到等等异常),请看代码实现片段:

#_*_ coding:utf-8 _*_ __author__ = '苦叶子' from selenium import webdriver if __name__ == '__main__': driver = webdrivrer.Firefox() driver.implicitly_wait(30) # 隐性等待,最长等30s driver.get('http://www.testingunion.com') print u"当前url: ", driver.current_url driver.quit()

注:上述代码中加粗删除线的代码通过调用webdriver提供的标准隐性等待方式来实现一种同步机制。其设置的是一个最长等待时间,如果在规定的时间未完成,则进入下一步。

不足:在实践中,通常我们需要操作的元素已经显示出来,但因网络或其他因素,浏览器一直处于加载个别js或图片或其他资源时,隐性等待模式下,这时会依旧处于等待状态直至页面全部加载完毕才能进入下一步。那有没有更好的办法呢?当然是有的,请参见下一方式。

重要:隐性等待是全局性质的,只需在driver实例化后,设置一次即可。

在实践中,经常见到新手把隐性等待当做sleep来使用,在每个步骤后都用一次。

三、显性等待

更为强大的方式是显性等待来实现同步机制,需要WebDriverWait类,辅以until()或until_not()方法,根据判断条件进行灵活的同步,它的主要机制是:程序在规定的时间内每个xx秒看一下判断条件是否成立,如果成立则执行下一步,否则继续等待,直至超过设置的最长时间,然后抛出异常。请看具体的代码片段:

#_*_ coding:utf-8 _*_ __author__ = '苦叶子' 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 if __name__ == '__main__': driver = webdrivrer.Firefox() # 隐性等待和显性等待可以同时用,要注意的是:最大等待时间取决两者之间的大值 driver.implicitly_wait(10) driver.get('http://www.testingunion.com') locator = (By.LINK_TEXT, u'webdriver') try: # 在最长20s内,每个0.5秒去检查locator是否存在,如果存在则进入下一步 WebDriverWait(driver, 20, 0.5).until(EC.presence_of_located(locator)) # 提取该文本对应的url,并打印出来 print driver.find_element_by_link_text(u'webdriver').get_attribute('href') finally: print u"异常了" driver.quit()

在本示例中,我们设置了隐性等待和显性等待,在其他的操作中隐性等待起决定性作用,在示例中的WebDriverWait设置了显性等待的地方,则显性等待起决定性作用,但要注意的是:最长等待时间取决于两者之间的大值,所以这里显性等待的最长时间为20s。

在这里我们主要用到了WebDriverWait类和expected_conditions模块,让我们近距离的接触下它们。

WebDriverWait类

定义实现在wait模块中,实现了webdriver的显性等待机制,先看下它有哪些参数和方法:

selenium.webdriver.support.wait.WebDriverWait(类) __init__(self, driver, # 传入实例化的webdriver对象 timeout, # 超时时间,等待的最大时间(需考虑同时考虑隐性等待时间) poll_frequency=POLL_FREQUENCY, # 调用until或until_not方法的间隔时间,上例为0.5s ignored_exceptions=None #指定忽略的异常,如果指定了要忽略的异常类型,则在调用until或until_not过程中,捕获该类异常时不中断代码,继续等待。默认只有NoSuchElementException )


until(self, method, # 在等待期间,每个一段时间调用这个传入的方法,直到返回值为false message='' # 如果超时,则抛出TimeoutException,将message传入给异常 ) until_not 与until相反,until是当某个元素满足某种条件时(出现、存在等等)继续执行;until_not则是当某个元素不满足某种条件时继续执行,参数含义相同

特别注意:

很多时候大家在使用until或until_not时,会将一个WebElement对象传入给method,如下:

WebDriverWait(driver, 10).until(driver.find_element_by_id('kw')) # 这是错误的用法

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

TypeError: 'xxx' object is not callable

在这里,也可以用selenium2提供的expected_conditions模块中提供的各种条件,也可用WebElement中的is_displayed(),is_selected(),is_enabled()方法或是自己封装的方法均可。下面我们再看看selenium2提供了哪些条件,如图所示:

结束语

本文就python selenium2三种同步解决方式进行了较为详细的说明,这是使用selenium2进行自动化测试实践的必备技能,希望对大家有所帮助,有任何问题请关注公众号号,直接回复消息进行交流。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2017-03-22,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 开源优测 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 一 强制等待
  • 二、隐性等待
  • 三、显性等待
  • WebDriverWait类
  • 结束语
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档