我們?cè)谑褂胹elenium的時(shí)候材蛛,很多時(shí)候會(huì)出現(xiàn)定位不到的時(shí)候苇瓣。在大多數(shù)情況下就是兩種問(wèn)題:1.有frame的問(wèn)題 2 沒(méi)有加等待條件剧浸。代碼運(yùn)行的速度和瀏覽器加載的速度不太相同画机。而解決這個(gè)問(wèn)題只有一個(gè)辦法,那就是等待系任。下面介紹三種等待的方法恳蹲。
1. 強(qiáng)制等待
第一種也是最簡(jiǎn)單粗暴的一種辦法就是強(qiáng)制等待sleep(xx),強(qiáng)制讓閃電俠等xx時(shí)間俩滥,不管凹凸曼能不能跟上速度嘉蕾,還是已經(jīng)提前到了,都必須等xx時(shí)間霜旧。
`# -*- coding: utf-8 -*-`
`from` `selenium` `import` `webdriver`
`from` `time` `import` `sleep`
`driver` `=` `webdriver.Firefox()`
`driver.get(``'[https://huilansame.github.io](https://huilansame.github.io/)'``)`
`sleep(``3``)` `# 強(qiáng)制等待3秒再執(zhí)行下一步`
`print` `driver.current_url`
`driver.quit()`
這種叫強(qiáng)制等待错忱,不管你瀏覽器是否加載完了,程序都得等待3秒挂据,3秒一到以清,繼續(xù)執(zhí)行下面的代碼,作為調(diào)試很有用崎逃,有時(shí)候也可以在代碼里這樣等待玖媚,不過(guò)不建議總用這種等待方式,太死板婚脱,嚴(yán)重影響程序執(zhí)行速度。
2. 隱性等待
第二種辦法叫隱性等待勺像,implicitly_wait(xx)障贸,隱性等待的意義是:閃電俠和凹凸曼約定好,不論閃電俠去哪兒吟宦,都要等凹凸曼xx秒篮洁,如果凹凸曼在這段時(shí)間內(nèi)來(lái)了,則倆人立即出發(fā)去打怪獸殃姓,如果凹凸曼在規(guī)定時(shí)間內(nèi)沒(méi)到袁波,則閃電俠自己去,那自然就等著凹凸曼給你拋異常吧蜗侈。
`# -*- coding: utf-8 -*-`
`from` `selenium` `import` `webdriver`
`driver` `=` `webdriver.Firefox()`
`driver.implicitly_wait(``30``)` `# 隱性等待篷牌,最長(zhǎng)等30秒`
`driver.get(``'[https://huilansame.github.io](https://huilansame.github.io/)'``)`
`print` `driver.current_url`
`driver.quit()`
隱形等待是設(shè)置了一個(gè)最長(zhǎng)等待時(shí)間,如果在規(guī)定時(shí)間內(nèi)網(wǎng)頁(yè)加載完成踏幻,則執(zhí)行下一步枷颊,否則一直等到時(shí)間截止,然后執(zhí)行下一步。注意這里有一個(gè)弊端夭苗,那就是程序會(huì)一直等待整個(gè)頁(yè)面加載完成信卡,也就是一般情況下你看到瀏覽器標(biāo)簽欄那個(gè)小圈不再轉(zhuǎn),才會(huì)執(zhí)行下一步题造,但有時(shí)候頁(yè)面想要的元素早就在加載完成了傍菇,但是因?yàn)閭€(gè)別js之類(lèi)的東西特別慢,我仍得等到頁(yè)面全部完成才能執(zhí)行下一步界赔,我想等我要的元素出來(lái)之后就下一步怎么辦丢习?有辦法,這就要看selenium提供的另一種等待方式——顯性等待wait了仔蝌。
需要特別說(shuō)明的是:隱性等待對(duì)整個(gè)driver的周期都起作用泛领,所以只要設(shè)置一次即可,我曾看到有人把隱性等待當(dāng)成了sleep在用敛惊,走哪兒都來(lái)一下…
3. 顯性等待
三種辦法就是顯性等待渊鞋,WebDriverWait,配合該類(lèi)的until()和until_not()方法瞧挤,就能夠根據(jù)判斷條件而進(jìn)行靈活地等待了锡宋。它主要的意思就是:程序每隔xx秒看一眼,如果條件成立了特恬,則執(zhí)行下一步执俩,否則繼續(xù)等待,直到超過(guò)設(shè)置的最長(zhǎng)時(shí)間癌刽,然后拋出TimeoutException役首。
`# -*- coding: utf-8 -*-`
`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`
`driver` `=` `webdriver.Firefox()`
`driver.implicitly_wait(``10``)` `# 隱性等待和顯性等待可以同時(shí)用,但要注意:等待的最長(zhǎng)時(shí)間取兩者之中的大者`
`driver.get(``'[https://huilansame.github.io](https://huilansame.github.io/)'``)`
`locator` `=` `(By.LINK_TEXT,` `'CSDN'``)`
`try``:`
`WebDriverWait(driver,` `20``,` `0.5``).until(EC.presence_of_element_located(locator))`
`print` `driver.find_element_by_link_text(``'CSDN'``).get_attribute(``'href'``)`
`finally``:`
`driver.close()`
上例中显拜,我們?cè)O(shè)置了隱性等待和顯性等待衡奥,在其他操作中,隱性等待起決定性作用远荠,在WebDriverWait..中顯性等待起主要作用矮固,但要注意的是:最長(zhǎng)的等待時(shí)間取決于兩者之間的大者,此例中為20譬淳,如果隱性等待時(shí)間 > 顯性等待時(shí)間档址,則該句代碼的最長(zhǎng)等待時(shí)間等于隱性等待時(shí)間。
我們主要用到了WebDriverWait類(lèi)與expected_conditions模塊邻梆,下面帶大家細(xì)看一下這兩個(gè)模塊:
WebDriverWait
wait模塊的WebDriverWait類(lèi)是顯性等待類(lèi)守伸,先看下它有哪些參數(shù)與方法:
selenium.webdriver.support.wait.WebDriverWait(類(lèi))
__init__
driver: 傳入WebDriver實(shí)例,即我們上例中的driver
timeout: 超時(shí)時(shí)間浦妄,等待的最長(zhǎng)時(shí)間(同時(shí)要考慮隱性等待時(shí)間)
poll_frequency: 調(diào)用until或until_not中的方法的間隔時(shí)間含友,默認(rèn)是0.5秒
ignored_exceptions: 忽略的異常替裆,如果在調(diào)用until或until_not的過(guò)程中拋出這個(gè)元組中的異常,
則不中斷代碼窘问,繼續(xù)等待辆童,如果拋出的是這個(gè)元組外的異常,則中斷代碼惠赫,拋出異常把鉴。默認(rèn)只有NoSuchElementException。
until
method: 在等待期間儿咱,每隔一段時(shí)間調(diào)用這個(gè)傳入的方法庭砍,直到返回值不是False
message: 如果超時(shí),拋出TimeoutException混埠,將message傳入異常
until_not 與until相反怠缸,until是當(dāng)某元素出現(xiàn)或什么條件成立則繼續(xù)執(zhí)行,
until_not是當(dāng)某元素消失或什么條件不成立則繼續(xù)執(zhí)行钳宪,參數(shù)也相同揭北,不再贅述。
method
message
看了以上內(nèi)容基本上很清楚了吏颖,調(diào)用方法如下:
WebDriverWait(driver, 超時(shí)時(shí)長(zhǎng), 調(diào)用頻率, 忽略異常).until(可執(zhí)行方法, 超時(shí)時(shí)返回的信息)
這里需要特別注意的是until或until_not中的可執(zhí)行方法method參數(shù)搔体,很多人傳入了WebElement對(duì)象,如下:
WebDriverWait(driver, 10).until(driver.find_element_by_id('kw')) # 錯(cuò)誤
這是錯(cuò)誤的用法半醉,這里的參數(shù)一定要是可以調(diào)用的疚俱,即這個(gè)對(duì)象一定有 call() 方法,否則會(huì)拋出異常:
TypeError: 'xxx' object is not callable
在這里缩多,你可以用selenium提供的 expected_conditions 模塊中的各種條件呆奕,也可以用WebElement的 is_displayed() 、is_enabled()衬吆、is_selected() 方法登馒,或者用自己封裝的方法都可以,那么接下來(lái)我們看一下selenium提供的條件有哪些expected_conditions:
expected_conditions
expected_conditions是selenium的一個(gè)模塊咆槽,其中包含一系列可用于判斷的條件:
selenium.webdriver.support.expected_conditions(模塊)
這兩個(gè)條件類(lèi)驗(yàn)證title,驗(yàn)證傳入的參數(shù)title是否等于或包含于driver.title
title_is
title_contains
這兩個(gè)人條件驗(yàn)證元素是否出現(xiàn)圈纺,傳入的參數(shù)都是元組類(lèi)型的locator秦忿,如(By.ID, 'kw')
顧名思義,一個(gè)只要一個(gè)符合條件的元素加載出來(lái)就通過(guò)蛾娶;另一個(gè)必須所有符合條件的元素都加載出來(lái)才行
presence_of_element_located
presence_of_all_elements_located
這三個(gè)條件驗(yàn)證元素是否可見(jiàn)灯谣,前兩個(gè)傳入?yún)?shù)是元組類(lèi)型的locator,第三個(gè)傳入WebElement
第一個(gè)和第三個(gè)其實(shí)質(zhì)是一樣的
visibility_of_element_located
invisibility_of_element_located
visibility_of
這兩個(gè)人條件判斷某段文本是否出現(xiàn)在某元素中蛔琅,一個(gè)判斷元素的text胎许,一個(gè)判斷元素的value
text_to_be_present_in_element
text_to_be_present_in_element_value
這個(gè)條件判斷frame是否可切入,可傳入locator元組或者直接傳入定位方式:id、name辜窑、index或WebElement
frame_to_be_available_and_switch_to_it
這個(gè)條件判斷是否有alert出現(xiàn)
alert_is_present
這個(gè)條件判斷元素是否可點(diǎn)擊钩述,傳入locator
element_to_be_clickable
這四個(gè)條件判斷元素是否被選中,第一個(gè)條件傳入WebElement對(duì)象穆碎,第二個(gè)傳入locator元組
第三個(gè)傳入WebElement對(duì)象以及狀態(tài)牙勘,相等返回True,否則返回False
第四個(gè)傳入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
最后一個(gè)條件判斷一個(gè)元素是否仍在DOM中,傳入WebElement對(duì)象色徘,可以判斷頁(yè)面是否刷新了
staleness_of
上面是所有17個(gè)condition恭金,與until、until_not組合能夠?qū)崿F(xiàn)很多判斷褂策,如果能自己靈活封裝横腿,將會(huì)大大提高腳本的穩(wěn)定性。