1. 程式人生 > >Selenium-三種等待方式

Selenium-三種等待方式

wrapper .get ui自動化 web開發 最長 chrome 一段 keys eve

在UI自動化測試中,必然會遇到環境不穩定,網絡慢的情況,這時如果不做任何處理的話,代碼會由於沒有找到元素而報錯。這時我們就要用到wait,而在Selenium中,我們可以用到一共三種等待,每一種等待都有自己的優點或缺點,如果選擇最優的等待方式。

time(固定等待)

在開發自動化框架過程中,最忌諱使用python自帶模塊的time的sleep方式進行等待,雖然可以自定義等待時間,但當網絡條件良好時,依舊按照預設定的時間繼續等待,導致整個項目的自動化時間無限延長。不建議使用。(註:腳本調試過程時,還是可以使用的,方便快捷)

implicitly_wait(隱式等待)

隱式等待實際是設置了一個最長等待時間,如果在規定時間內網頁加載完成,則執行下一步,否則一直等到時間結束,然後執行下一步。這樣的隱式等待會有個坑,我們都知道js一般都是放在我們的body的最後進行加載,實際這是頁面上的元素都已經加載完畢,我們卻還在等待全部頁面加載結束。隱式等待對整個driver周期都起作用,在最開始設置一次就可以了。不要當作固定等待使用,到哪都來一下隱式等待。

WebDriverWait(顯示等待)

WebDriverWait是selenium提供得到顯示等待模塊引入路徑

from selenium.webdriver.support.wait import WebDriver

WebDriverWait參數

driver: 傳入WebDriver實例,即我們上例中的driver
timeout: 超時時間,等待的最長時間
poll_frequency: 調用until或until_not中的方法的間隔時間,默認是0.5秒
ignored_exceptions: 忽略的異常,如果在調用until或until_not的過程中拋出這個元組中的異常,
則不中斷代碼,繼續等待,如果拋出的是這個元組外的異常,則中斷代碼,拋出異常。默認只有NoSuchElementException。

這個模塊中,一共只有兩種方法until與until_not

method: 在等待期間,每隔一段時間調用這個傳入的方法,直到返回值不是False
message: 如果超時,拋出TimeoutException,將message傳入異常

until

當某元素出現或什麽條件成立則繼續執行

until_not

當某元素消失或什麽條件不成立則繼續執行

兩個方法的method,必須是含有__call__的可執行方法。所以我們引用selenium提供的一個模塊

from selenium.webdriver.support import expected_conditions as Ec
‘‘‘
隱式等待和顯示等待都存在時,超時時間取二者中較大的‘‘‘ locator = (By.ID,kw) driver.get(base_url) WebDriverWait(driver,10).until(EC.title_is(u"百度一下,你就知道")) ‘‘‘判斷title,返回布爾值‘‘‘ WebDriverWait(driver,10).until(EC.title_contains(u"百度一下")) ‘‘‘判斷title,返回布爾值‘‘‘ WebDriverWait(driver,10).until(EC.presence_of_element_located((By.ID,kw))) ‘‘‘判斷某個元素是否被加到了dom樹裏,並不代表該元素一定可見,如果定位到就返回WebElement‘‘‘ WebDriverWait(driver,10).until(EC.visibility_of_element_located((By.ID,su))) ‘‘‘判斷某個元素是否被添加到了dom裏並且可見,可見代表元素可顯示且寬和高都大於0‘‘‘ WebDriverWait(driver,10).until(EC.visibility_of(driver.find_element(by=By.ID,value=kw))) ‘‘‘判斷元素是否可見,如果可見就返回這個元素‘‘‘ WebDriverWait(driver,10).until(EC.presence_of_all_elements_located((By.CSS_SELECTOR,.mnav))) ‘‘‘判斷是否至少有1個元素存在於dom樹中,如果定位到就返回列表‘‘‘ WebDriverWait(driver,10).until(EC.visibility_of_any_elements_located((By.CSS_SELECTOR,.mnav))) ‘‘‘判斷是否至少有一個元素在頁面中可見,如果定位到就返回列表‘‘‘ WebDriverWait(driver,10).until(EC.text_to_be_present_in_element((By.XPATH,"//*[@id=‘u1‘]/a[8]"),u設置)) ‘‘‘判斷指定的元素中是否包含了預期的字符串,返回布爾值‘‘‘ WebDriverWait(driver,10).until(EC.text_to_be_present_in_element_value((By.CSS_SELECTOR,#su),u百度一下)) ‘‘‘判斷指定元素的屬性值中是否包含了預期的字符串,返回布爾值‘‘‘ #WebDriverWait(driver,10).until(EC.frame_to_be_available_and_switch_to_it(locator)) ‘‘‘判斷該frame是否可以switch進去,如果可以的話,返回True並且switch進去,否則返回False‘‘‘ #註意這裏並沒有一個frame可以切換進去 WebDriverWait(driver,10).until(EC.invisibility_of_element_located((By.CSS_SELECTOR,#swfEveryCookieWrap))) ‘‘‘判斷某個元素在是否存在於dom或不可見,如果可見返回False,不可見返回這個元素‘‘‘ #註意#swfEveryCookieWrap在此頁面中是一個隱藏的元素 WebDriverWait(driver,10).until(EC.element_to_be_clickable((By.XPATH,"//*[@id=‘u1‘]/a[8]"))).click() ‘‘‘判斷某個元素中是否可見並且是enable的,代表可點擊‘‘‘ driver.find_element_by_xpath("//*[@id=‘wrapper‘]/div[6]/a[1]").click() #WebDriverWait(driver,10).until(EC.element_to_be_clickable((By.XPATH,"//*[@id=‘wrapper‘]/div[6]/a[1]"))).click() #WebDriverWait(driver,10).until(EC.staleness_of(driver.find_element(By.ID,‘su‘))) ‘‘‘等待某個元素從dom樹中移除‘‘‘ #這裏沒有找到合適的例子 WebDriverWait(driver,10).until(EC.element_to_be_selected(driver.find_element(By.XPATH,"//*[@id=‘nr‘]/option[1]"))) ‘‘‘判斷某個元素是否被選中了,一般用在下拉列表‘‘‘ WebDriverWait(driver,10).until(EC.element_selection_state_to_be(driver.find_element(By.XPATH,"//*[@id=‘nr‘]/option[1]"),True)) ‘‘‘判斷某個元素的選中狀態是否符合預期‘‘‘ WebDriverWait(driver,10).until(EC.element_located_selection_state_to_be((By.XPATH,"//*[@id=‘nr‘]/option[1]"),True)) ‘‘‘判斷某個元素的選中狀態是否符合預期‘‘‘ driver.find_element_by_xpath(".//*[@id=‘gxszButton‘]/a[1]").click() instance = WebDriverWait(driver,10).until(EC.alert_is_present()) ‘‘‘判斷頁面上是否存在alert,如果有就切換到alert並返回alert的內容‘‘‘ print instance.text instance.accept()
from selenium import webdriver
driver=webdriver.Chrome()
driver.get("http://ui.imdsx.cn/uitester/")
from selenium.webdriver.support.wait import WebDriverWait
from selenium.webdriver.support import expected_conditions as Ec
driver.maximize_window()#將窗口放大
driver.execute_script(window.scrollTo(0,0);)
e=WebDriverWait(driver,10).until(Ec.presence_of_element_located((id,i1)))#等什麽出現則繼續執行
e.send_keys(1111)
from selenium.webdriver.common.by import By
e=WebDriverWait(driver,10).until(Ec.presence_of_element_located((By.ID,i1)))
WebDriverWait(driver,10,1)10:超長時間。1:步長,代表每1秒查詢一次
WebDriverWait(driver,10).until_not()等什麽消失繼續執行
UI自動化的弊端
1.維護成本大(產量不高)
    a.代碼沒做任何的解耦
    b.沒有框架的思想,純粹在堆代碼(pageobject)
        根據項目的每個頁面,抽象出類,每個功能點抽象出這個類的函數
    c.web開發會變更html接口,或進行修改
        css_selector的定位方式,盡量減輕層級定位的使用次數
2.代碼愛報錯
    WebDriverWait每隔幾秒鐘掃描一次,如果超時了,則報timeout的錯誤
    css_selector的定位方式,盡量減輕層級定位的使用次數

Selenium-三種等待方式