很多人在群里問慕嚷,這個下拉框定位不到、那個彈出框定位不到…各種定位不到莺掠,其實大多數(shù)情況下就是兩種問題:1 有frame衫嵌,2 沒有加等待。殊不知彻秆,你的代碼運行速度是什么量級的楔绞,而瀏覽器加載渲染速度又是什么量級的,就好比閃電俠和凹凸曼約好去打怪獸唇兑,然后閃電俠打完回來之后問凹凸曼你為啥還在穿鞋沒出門酒朵?凹凸曼分分中內(nèi)心一萬只羊駝飛過,欺負(fù)哥速度慢扎附,哥不跟你玩了蔫耽,拋個異常撂挑子了。
那么怎么才能照顧到凹凸曼緩慢的加載速度呢留夜?只有一個辦法匙铡,那就是等嘍。說到等碍粥,又有三種等法鳖眼,且聽博主一一道來:
1. 強制等待
第一種也是最簡單粗暴的一種辦法就是強制等待sleep(xx),強制讓閃電俠等xx時間嚼摩,不管凹凸曼能不能跟上速度钦讳,還是已經(jīng)提前到了,都必須等xx時間枕面。
看代碼:
# -*- coding: utf-8 -*-
fromselenium importwebdriver
fromtime importsleep
driver =webdriver.Firefox()
driver.get('https://huilansame.github.io')
sleep(3) # 強制等待3秒再執(zhí)行下一步
printdriver.current_url
driver.quit()
這種叫強制等待愿卒,不管你瀏覽器是否加載完了,程序都得等待3秒潮秘,3秒一到琼开,繼續(xù)執(zhí)行下面的代碼,作為調(diào)試很有用枕荞,有時候也可以在代碼里這樣等待稠通,不過不建議總用這種等待方式,太死板买猖,嚴(yán)重影響程序執(zhí)行速度。
2. 隱性等待
第二種辦法叫隱性等待滋尉,implicitly_wait(xx)玉控,隱性等待的意義是:閃電俠和凹凸曼約定好,不論閃電俠去哪兒狮惜,都要等凹凸曼xx秒高诺,如果凹凸曼在這段時間內(nèi)來了碌识,則倆人立即出發(fā)去打怪獸,如果凹凸曼在規(guī)定時間內(nèi)沒到虱而,則閃電俠自己去筏餐,那自然就等著凹凸曼給你拋異常吧。
看代碼:
# -*- coding: utf-8 -*-
fromselenium importwebdriver
driver =webdriver.Firefox()
driver.implicitly_wait(30) # 隱性等待牡拇,最長等30秒
driver.get('https://huilansame.github.io')
printdriver.current_url
driver.quit()
隱形等待是設(shè)置了一個最長等待時間魁瞪,如果在規(guī)定時間內(nèi)網(wǎng)頁加載完成,則執(zhí)行下一步惠呼,否則一直等到時間截止导俘,然后執(zhí)行下一步。注意這里有一個弊端剔蹋,那就是程序會一直等待整個頁面加載完成旅薄,也就是一般情況下你看到瀏覽器標(biāo)簽欄那個小圈不再轉(zhuǎn),才會執(zhí)行下一步泣崩,但有時候頁面想要的元素早就在加載完成了少梁,但是因為個別js之類的東西特別慢,我仍得等到頁面全部完成才能執(zhí)行下一步矫付,我想等我要的元素出來之后就下一步怎么辦凯沪?有辦法,這就要看selenium提供的另一種等待方式——顯性等待wait了技即。
需要特別說明的是:隱性等待對整個driver的周期都起作用著洼,所以只要設(shè)置一次即可,我曾看到有人把隱性等待當(dāng)成了sleep在用而叼,走哪兒都來一下…
3. 顯性等待
第三種辦法就是顯性等待身笤,WebDriverWait,配合該類的until()和until_not()方法葵陵,就能夠根據(jù)判斷條件而進行靈活地等待了液荸。它主要的意思就是:程序每隔xx秒看一眼,如果條件成立了脱篙,則執(zhí)行下一步娇钱,否則繼續(xù)等待,直到超過設(shè)置的最長時間绊困,然后拋出TimeoutException文搂。
先看個代碼示例:
# -*- coding: utf-8 -*-
fromselenium importwebdriver
fromselenium.webdriver.support.wait importWebDriverWait
fromselenium.webdriver.support importexpected_conditions as EC
fromselenium.webdriver.common.by importBy
driver =webdriver.Firefox()
driver.implicitly_wait(10) # 隱性等待和顯性等待可以同時用,但要注意:等待的最長時間取兩者之中的大者
driver.get('https://huilansame.github.io')
locator =(By.LINK_TEXT, 'CSDN')
try:
??WebDriverWait(driver, 20, 0.5).until(EC.presence_of_element_located(locator))
??printdriver.find_element_by_link_text('CSDN').get_attribute('href')
finally:
??driver.close()
上例中秤朗,我們設(shè)置了隱性等待和顯性等待煤蹭,在其他操作中,隱性等待起決定性作用,在WebDriverWait..中顯性等待起主要作用硝皂,但要注意的是:最長的等待時間取決于兩者之間的大者常挚,此例中為20,如果隱性等待時間 > 顯性等待時間稽物,則該句代碼的最長等待時間等于隱性等待時間奄毡。
我們主要用到了WebDriverWait類與expected_conditions模塊,下面博主帶大家細(xì)看一下這兩個模塊:
WebDriverWait
wait模塊的WebDriverWait類是顯性等待類贝或,先看下它有哪些參數(shù)與方法:
selenium.webdriver.support.wait.WebDriverWait(類)
__init__
??driver: 傳入WebDriver實例吼过,即我們上例中的driver
??timeout: 超時時間,等待的最長時間(同時要考慮隱性等待時間)
??poll_frequency: 調(diào)用until或until_not中的方法的間隔時間傀缩,默認(rèn)是0.5秒
??ignored_exceptions: 忽略的異常那先,如果在調(diào)用until或until_not的過程中拋出這個元組中的異常,
??????則不中斷代碼赡艰,繼續(xù)等待售淡,如果拋出的是這個元組外的異常,則中斷代碼慷垮,拋出異常揖闸。默認(rèn)只有NoSuchElementException。
until
??method: 在等待期間料身,每隔一段時間調(diào)用這個傳入的方法汤纸,直到返回值不是False
??message: 如果超時,拋出TimeoutException芹血,將message傳入異常
until_not 與until相反贮泞,until是當(dāng)某元素出現(xiàn)或什么條件成立則繼續(xù)執(zhí)行,
????until_not是當(dāng)某元素消失或什么條件不成立則繼續(xù)執(zhí)行幔烛,參數(shù)也相同啃擦,不再贅述。
看了以上內(nèi)容基本上很清楚了饿悬,調(diào)用方法如下:
WebDriverWait(driver, 超時時長, 調(diào)用頻率, 忽略異常).until(可執(zhí)行方法, 超時時返回的信息)
這里需要特別注意的是until或until_not中的可執(zhí)行方法method參數(shù)令蛉,很多人傳入了WebElement對象,如下:
WebDriverWait(driver, 10).until(driver.find_element_by_id('kw'))? # 錯誤
這是錯誤的用法狡恬,這里的參數(shù)一定要是可以調(diào)用的珠叔,即這個對象一定有 __call__() 方法,否則會拋出異常:
TypeError: 'xxx' object is not callable
可用弟劲,WebDriverWait(driver,10).until\(lambda x:x.find_element_by_id("kw")).\send_keys(u"劉德華“)祷安??
在這里兔乞,你可以用selenium提供的 expected_conditions 模塊中的各種條件汇鞭,也可以用WebElement的 is_displayed() 撇眯、is_enabled()、is_selected() 方法虱咧,或者用自己封裝的方法都可以,那么接下來我們看一下selenium提供的條件有哪些:
expected_conditions? 期望事件锚国,要引入EC
expected_conditions是selenium的一個模塊腕巡,其中包含一系列可用于判斷的條件:
selenium.webdriver.support.expected_conditions(模塊)
或者:selenium.webdriver.support.expected_conditions as EC
這兩個條件類驗證title,驗證傳入的參數(shù)title是否等于或包含于driver.title
title_is
title_contains
這兩個人條件驗證元素是否出現(xiàn)血筑,傳入的參數(shù)都是元組類型的locator绘沉,如(By.ID, 'kw')
顧名思義,一個只要一個符合條件的元素加載出來就通過豺总;另一個必須所有符合條件的元素都加載出來才行
presence_of_element_located
presence_of_all_elements_located
這三個條件驗證元素是否可見车伞,前兩個傳入?yún)?shù)是元組類型的locator,第三個傳入WebElement
第一個和第三個其實質(zhì)是一樣的
visibility_of_element_located
invisibility_of_element_located
visibility_of
這兩個人條件判斷某段文本是否出現(xiàn)在某元素中喻喳,一個判斷元素的text另玖,一個判斷元素的value
text_to_be_present_in_element
text_to_be_present_in_element_value
這個條件判斷frame是否可切入,可傳入locator元組或者直接傳入定位方式:id表伦、name谦去、index或WebElement
frame_to_be_available_and_switch_to_it
這個條件判斷是否有alert出現(xiàn)
alert_is_present
這個條件判斷元素是否可點擊,傳入locator
element_to_be_clickable
這四個條件判斷元素是否被選中蹦哼,第一個條件傳入WebElement對象鳄哭,第二個傳入locator元組
第三個傳入WebElement對象以及狀態(tài),相等返回True纲熏,否則返回False
第四個傳入locator以及狀態(tài)妆丘,相等返回True,否則返回False
element_to_be_selected
element_located_to_be_selected
element_selection_state_to_be
element_located_selection_state_to_be
最后一個條件判斷一個元素是否仍在DOM中局劲,傳入WebElement對象勺拣,可以判斷頁面是否刷新了
staleness_of
上面是所有17個condition,與until容握、until_not組合能夠?qū)崿F(xiàn)很多判斷宣脉,如果能自己靈活封裝,將會大大提高腳本的穩(wěn)定性剔氏。