Selenium-三种等待方式
在UI自動(dòng)化測(cè)試中,必然會(huì)遇到環(huán)境不穩(wěn)定,網(wǎng)絡(luò)慢的情況,這時(shí)如果不做任何處理的話(huà),代碼會(huì)由于沒(méi)有找到元素而報(bào)錯(cuò)。這時(shí)我們就要用到wait,而在Selenium中,我們可以用到一共三種等待,每一種等待都有自己的優(yōu)點(diǎn)或缺點(diǎn),如果選擇最優(yōu)的等待方式。
time(固定等待)
在開(kāi)發(fā)自動(dòng)化框架過(guò)程中,最忌諱使用python自帶模塊的time的sleep方式進(jìn)行等待,雖然可以自定義等待時(shí)間,但當(dāng)網(wǎng)絡(luò)條件良好時(shí),依舊按照預(yù)設(shè)定的時(shí)間繼續(xù)等待,導(dǎo)致整個(gè)項(xiàng)目的自動(dòng)化時(shí)間無(wú)限延長(zhǎng)。不建議使用。(注:腳本調(diào)試過(guò)程時(shí),還是可以使用的,方便快捷)
implicitly_wait(隱式等待)
隱式等待實(shí)際是設(shè)置了一個(gè)最長(zhǎng)等待時(shí)間,如果在規(guī)定時(shí)間內(nèi)網(wǎng)頁(yè)加載完成,則執(zhí)行下一步,否則一直等到時(shí)間結(jié)束,然后執(zhí)行下一步。這樣的隱式等待會(huì)有個(gè)坑,我們都知道js一般都是放在我們的body的最后進(jìn)行加載,實(shí)際這是頁(yè)面上的元素都已經(jīng)加載完畢,我們卻還在等待全部頁(yè)面加載結(jié)束。隱式等待對(duì)整個(gè)driver周期都起作用,在最開(kāi)始設(shè)置一次就可以了。不要當(dāng)作固定等待使用,到哪都來(lái)一下隱式等待。
WebDriverWait(顯示等待)
WebDriverWait是selenium提供得到顯示等待模塊引入路徑
from selenium.webdriver.support.wait import WebDriverWaitWebDriverWait參數(shù)
driver: 傳入WebDriver實(shí)例,即我們上例中的driver timeout: 超時(shí)時(shí)間,等待的最長(zhǎng)時(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。這個(gè)模塊中,一共只有兩種方法until與until_not
method: 在等待期間,每隔一段時(shí)間調(diào)用這個(gè)傳入的方法,直到返回值不是False message: 如果超時(shí),拋出TimeoutException,將message傳入異常until
當(dāng)某元素出現(xiàn)或什么條件成立則繼續(xù)執(zhí)行
until_not
當(dāng)某元素消失或什么條件不成立則繼續(xù)執(zhí)行
兩個(gè)方法的method,必須是含有__call__的可執(zhí)行方法。所以我們引用selenium提供的一個(gè)模塊
from selenium.webdriver.support import expected_conditions as Ec '''隱式等待和顯示等待都存在時(shí),超時(shí)時(shí)間取二者中較大的''' 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'))) '''判斷某個(gè)元素是否被加到了dom樹(shù)里,并不代表該元素一定可見(jiàn),如果定位到就返回WebElement'''WebDriverWait(driver,10).until(EC.visibility_of_element_located((By.ID,'su'))) '''判斷某個(gè)元素是否被添加到了dom里并且可見(jiàn),可見(jiàn)代表元素可顯示且寬和高都大于0'''WebDriverWait(driver,10).until(EC.visibility_of(driver.find_element(by=By.ID,value='kw'))) '''判斷元素是否可見(jiàn),如果可見(jiàn)就返回這個(gè)元素'''WebDriverWait(driver,10).until(EC.presence_of_all_elements_located((By.CSS_SELECTOR,'.mnav'))) '''判斷是否至少有1個(gè)元素存在于dom樹(shù)中,如果定位到就返回列表'''WebDriverWait(driver,10).until(EC.visibility_of_any_elements_located((By.CSS_SELECTOR,'.mnav'))) '''判斷是否至少有一個(gè)元素在頁(yè)面中可見(jiàn),如果定位到就返回列表'''WebDriverWait(driver,10).until(EC.text_to_be_present_in_element((By.XPATH,"//*[@id='u1']/a[8]"),u'設(shè)置')) '''判斷指定的元素中是否包含了預(yù)期的字符串,返回布爾值'''WebDriverWait(driver,10).until(EC.text_to_be_present_in_element_value((By.CSS_SELECTOR,'#su'),u'百度一下')) '''判斷指定元素的屬性值中是否包含了預(yù)期的字符串,返回布爾值'''#WebDriverWait(driver,10).until(EC.frame_to_be_available_and_switch_to_it(locator)) '''判斷該frame是否可以switch進(jìn)去,如果可以的話(huà),返回True并且switch進(jìn)去,否則返回False''' #注意這里并沒(méi)有一個(gè)frame可以切換進(jìn)去 WebDriverWait(driver,10).until(EC.invisibility_of_element_located((By.CSS_SELECTOR,'#swfEveryCookieWrap'))) '''判斷某個(gè)元素在是否存在于dom或不可見(jiàn),如果可見(jiàn)返回False,不可見(jiàn)返回這個(gè)元素''' #注意#swfEveryCookieWrap在此頁(yè)面中是一個(gè)隱藏的元素 WebDriverWait(driver,10).until(EC.element_to_be_clickable((By.XPATH,"//*[@id='u1']/a[8]"))).click() '''判斷某個(gè)元素中是否可見(jiàn)并且是enable的,代表可點(diǎn)擊''' 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'))) '''等待某個(gè)元素從dom樹(shù)中移除''' #這里沒(méi)有找到合適的例子 WebDriverWait(driver,10).until(EC.element_to_be_selected(driver.find_element(By.XPATH,"//*[@id='nr']/option[1]"))) '''判斷某個(gè)元素是否被選中了,一般用在下拉列表'''WebDriverWait(driver,10).until(EC.element_selection_state_to_be(driver.find_element(By.XPATH,"//*[@id='nr']/option[1]"),True)) '''判斷某個(gè)元素的選中狀態(tài)是否符合預(yù)期'''WebDriverWait(driver,10).until(EC.element_located_selection_state_to_be((By.XPATH,"//*[@id='nr']/option[1]"),True)) '''判斷某個(gè)元素的選中狀態(tài)是否符合預(yù)期''' driver.find_element_by_xpath(".//*[@id='gxszButton']/a[1]").click()instance = WebDriverWait(driver,10).until(EC.alert_is_present()) '''判斷頁(yè)面上是否存在alert,如果有就切換到alert并返回alert的內(nèi)容''' 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')))#等什么出現(xiàn)則繼續(xù)執(zhí)行 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:超長(zhǎng)時(shí)間。1:步長(zhǎng),代表每1秒查詢(xún)一次 WebDriverWait(driver,10).until_not()等什么消失繼續(xù)執(zhí)行 UI自動(dòng)化的弊端 1.維護(hù)成本大(產(chǎn)量不高)a.代碼沒(méi)做任何的解耦b.沒(méi)有框架的思想,純粹在堆代碼(pageobject)根據(jù)項(xiàng)目的每個(gè)頁(yè)面,抽象出類(lèi),每個(gè)功能點(diǎn)抽象出這個(gè)類(lèi)的函數(shù)c.web開(kāi)發(fā)會(huì)變更html接口,或進(jìn)行修改css_selector的定位方式,盡量減輕層級(jí)定位的使用次數(shù) 2.代碼愛(ài)報(bào)錯(cuò)WebDriverWait每隔幾秒鐘掃描一次,如果超時(shí)了,則報(bào)timeout的錯(cuò)誤css_selector的定位方式,盡量減輕層級(jí)定位的使用次數(shù)?
轉(zhuǎn)載于:https://www.cnblogs.com/wxcx/p/8948280.html
總結(jié)
以上是生活随笔為你收集整理的Selenium-三种等待方式的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: SpingBoot-Thymeleaf-
- 下一篇: vivo X21低调奢华 彭于晏携手黑金