Python selenium 三種等待方式解讀

1. 強制等待git

第一種也是最簡單粗暴的一種辦法就是強制等待sleep(xx),強制讓閃電俠等xx時間,無論凹凸曼能不能跟上速度,仍是已經提早到了,都必須等xx時間。github

看代碼:web

?
1
2
3
4
5
6
7
8
9
10
11
# -*- coding: utf-8 -*-
from selenium import webdriver
from time import sleep
 
driver = webdriver.Firefox()
 
sleep( 3 ) # 強制等待3秒再執行下一步
 
print driver.current_url
driver.quit()

這種叫強制等待,無論你瀏覽器是否加載完了,程序都得等待3秒,3秒一到,繼續執行下面的代碼,做爲調試頗有用,有時候也能夠在代碼裏這樣等待,不過不建議總用這種等待方式,太死板,嚴重影響程序執行速度。瀏覽器

2. 隱性等待ui

第二種辦法叫隱性等待,implicitly_wait(xx),隱性等待的意義是:閃電俠和凹凸曼約定好,不論閃電俠去哪兒,都要等凹凸曼xx秒,若是凹凸曼在這段時間內來了,則倆人當即出發去打怪獸,若是凹凸曼在規定時間內沒到,則閃電俠本身去,那天然就等着凹凸曼給你拋異常吧。url

看代碼:spa

?
1
2
3
4
5
6
7
8
9
# -*- coding: utf-8 -*-
from selenium import webdriver
 
driver = webdriver.Firefox()
driver.implicitly_wait( 30 ) # 隱性等待,最長等30秒
 
print driver.current_url
driver.quit()

隱形等待是設置了一個最長等待時間,若是在規定時間內網頁加載完成,則執行下一步,不然一直等到時間截止,而後執行下一步。注意這裏有一個弊端,那就是程序會一直等待整個頁面加載完成,也就是通常狀況下你看到瀏覽器標籤欄那個小圈再也不轉,纔會執行下一步,但有時候頁面想要的元素早就在加載完成了,可是由於個別js之類的東西特別慢,我仍得等到頁面所有完成才能執行下一步,我想等我要的元素出來以後就下一步怎麼辦?有辦法,這就要看selenium提供的另外一種等待方式——顯性等待wait了。.net

須要特別說明的是:隱性等待對整個driver的週期都起做用,因此只要設置一次便可,我曾看到有人把隱性等待當成了sleep在用,走哪兒都來一下…調試

3. 顯性等待code

第三種辦法就是顯性等待,WebDriverWait,配合該類的until()和until_not()方法,就可以根據判斷條件而進行靈活地等待了。它主要的意思就是:程序每隔xx秒看一眼,若是條件成立了,則執行下一步,不然繼續等待,直到超過設置的最長時間,而後拋出TimeoutException。

先看個代碼示例:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
# -*- 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 ) # 隱性等待和顯性等待能夠同時用,但要注意:等待的最長時間取二者之中的大者
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()

上例中,咱們設置了隱性等待和顯性等待,在其餘操做中,隱性等待起決定性做用,在WebDriverWait..中顯性等待起主要做用,但要注意的是:最長的等待時間取決於二者之間的大者,此例中爲20,若是隱性等待時間 > 顯性等待時間,則該句代碼的最長等待時間等於隱性等待時間。

咱們主要用到了WebDriverWait類與expected_conditions模塊,下面博主帶你們細看一下這兩個模塊:

WebDriverWait

wait模塊的WebDriverWait類是顯性等待類,先看下它有哪些參數與方法:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
selenium.webdriver.support.wait.WebDriverWait(類)
 
__init__
   driver: 傳入WebDriver實例,即咱們上例中的driver
   timeout: 超時時間,等待的最長時間(同時要考慮隱性等待時間)
   poll_frequency: 調用until或until_not中的方法的間隔時間,默認是 0.5
   ignored_exceptions: 忽略的異常,若是在調用until或until_not的過程當中拋出這個元組中的異常,
       則不中斷代碼,繼續等待,若是拋出的是這個元組外的異常,則中斷代碼,拋出異常。默認只有NoSuchElementException。
 
until
   method: 在等待期間,每隔一段時間調用這個傳入的方法,直到返回值不是 False
   message: 若是超時,拋出TimeoutException,將message傳入異常
 
until_not 與until相反,until是當某元素出現或什麼條件成立則繼續執行,
     until_not是當某元素消失或什麼條件不成立則繼續執行,參數也相同,再也不贅述。
   method
   message

看了以上內容基本上很清楚了,調用方法以下:

WebDriverWait(driver, 超時時長, 調用頻率, 忽略異常).until(可執行方法, 超時時返回的信息)

這裏須要特別注意的是until或until_not中的可執行方法method參數,不少人傳入了WebElement對象,以下:

WebDriverWait(driver, 10).until(driver.find_element_by_id('kw'))  # 錯誤

這是錯誤的用法,這裏的參數必定要是能夠調用的,即這個對象必定有 __call__() 方法,不然會拋出異常:

TypeError: 'xxx' object is not callable

在這裏,你能夠用selenium提供的 expected_conditions 模塊中的各類條件,也能夠用WebElement的 is_displayed() 、is_enabled()、is_selected() 方法,或者用本身封裝的方法均可以,那麼接下來咱們看一下selenium提供的條件有哪些:

expected_conditions

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
expected_conditions是selenium的一個模塊,其中包含一系列可用於判斷的條件:
 
selenium.webdriver.support.expected_conditions(模塊)
 
這兩個條件類驗證title,驗證傳入的參數title是否等於或包含於driver.title
title_is
title_contains
 
這兩我的條件驗證元素是否出現,傳入的參數都是元組類型的locator,如(By. ID , 'kw' )
顧名思義,一個只要一個符合條件的元素加載出來就經過;另外一個必須全部符合條件的元素都加載出來才行
presence_of_element_located
presence_of_all_elements_located
 
這三個條件驗證元素是否可見,前兩個傳入參數是元組類型的locator,第三個傳入WebElement
第一個和第三個其實質是同樣的
visibility_of_element_located
invisibility_of_element_located
visibility_of
 
這兩我的條件判斷某段文本是否出如今某元素中,一個判斷元素的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出現
alert_is_present
 
這個條件判斷元素是否可點擊,傳入locator
element_to_be_clickable
 
這四個條件判斷元素是否被選中,第一個條件傳入WebElement對象,第二個傳入locator元組
第三個傳入WebElement對象以及狀態,相等返回 True ,不然返回 False
第四個傳入locator以及狀態,相等返回 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組合可以實現不少判斷,若是能本身靈活封裝,將會大大提升腳本的穩定性。

相關文章
相關標籤/搜索