zoukankan      html  css  js  c++  java
  • 同步机制

    前言

    在使用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提供了哪些条件,如图所示:

     


    作者:苦叶子
    链接:https://www.jianshu.com/p/bb2c8f14dd3f
    來源:简书
    著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
  • 相关阅读:
    elasticsearch7.16.2集群部署
    MySQL on lion
    2012年10月 小记
    2013年01月小记
    配置tomcat+apache
    Maven安装
    android maven lion
    2012年11月小记
    Python:sys.argv[]的用法<转>
    Python:socket初学总结
  • 原文地址:https://www.cnblogs.com/x666-6/p/8391726.html
Copyright © 2011-2022 走看看