selenium(11):元素定位實際遇到的各種問題(持續更新)


1、class屬性有空格  (已驗證)

當classname 中存在空格的時候,直接使用find_element_by_class_name時,會顯示定位失敗,此時,需要將classname中的空格替換成英文的點“.”,如下圖:

 

 

 

此時如果使用 find_element_by_class_name("link detail"),則提示無法定位該元素

如果使用英文點替換,則可以正常定位 find_element_by_class_name("link.detail")

如果class有多個值,中間用空格分開,可以只用其中一個值(如果唯一)進行定位。 比如find_element_by_class_name("link")

2、上傳文件

用AutoIt上傳比較麻煩,找到上傳的input標簽,就可以通過send_keys()實現上傳:將本地文件的路徑作為一個值放在input標簽中,通過form表單將這個值提交給服務器。

 

如上圖所示,定位到文件輸入框,對應的是位置1 ,但是真正上傳的input標簽是位置2  id=upfile(上傳文件的意思),腳本如下:

find_element((By.ID,'upfile')).send_keys(file_path)

3、文件下載

方法一:(不推薦使用)

1、設置火狐瀏覽器自動下載到默認路徑,不跳出確認框。

2、找到瀏覽器配置文件夾路徑:幫助菜單—故障排除信息——配置文件夾-打開文件夾——獲取文件夾路徑fp

3、driver=webdriver.Firefox(firefox_profile=fp)  (不加firefox_profile參數打開的火狐是原生的瀏覽器,很干凈什么插件都沒有)

這時自動化腳本打開的就是實際使用的火狐,如果收藏地址、插件較多的話,打開會很慢。

方法二:(推薦) 

通過設置瀏覽器參數

option = webdriver.ChromeOptions()
prefs = {'profile.default_content_settings.popups': 0, 'download.default_directory': export_path} # 設置下載路徑,關閉確認彈出框。
option.add_experimental_option('prefs', prefs)
driver = webdriver.Chrome(chrome_options=option)

以上是chrome的設置方法,firefox類似。

4、/.. 定位  返回上級節點  和parent 效果一樣

 

 

方法1://td[@title='admin']/parent::tr

方法2://td[@title='admin']/..     更簡單明了  

5、following 結合 / 、// 的用法深入分析

following: 選取文檔中 當前節點的結束標簽之后 的所有節點,除屬性節點(節點中的屬性)命名空間節點(類似這樣的節點<div> </div>??)之外,如果當前節點有子節點,則包含其所有子節點。

xxx / following:: 選取xxx結束標簽之后的所有節點,不包括xxx的所有子節點。  '/' 表示從xxx后進行定位,
xxx // following:: 選取xxx結束標簽之后的所有節點, 包括xxx的所有子節點'//' 表示可以從xxx 任意一級子節點進行定位,以及xxx節點后進行定位。

如上圖 想定位到第一個input節點

 

 

 

定位1://div[@id='History_Table']/following::input[1]  

結果:沒有匹配元素,

原因:需要定位input屬於當前節點div[@id='History_Table']的子節點,而/ following 不包括當前節點的子節點。

 

定位2://div[@id='History_Table']//following::input[1]

結果:匹配到了當前節點中3個input子節點。

原因:// following  從當前節點后的任意一級子節點進行定位,所以3個都能匹配到。

 //div[@id='History_Table']//following::input[2]  則匹配后2個

//div[@id='History_Table']//following::input[3]  則匹配最后一個

 

定位3://div[@id='History_Table']/descendant::input[1]

 結果:成功定位到了第一個input

原因:descendant 選取當前節點的所有后代元素,/ descendant 從當前節點后進行定位,所以唯一

如果改成 //div[@id='History_Table']//descendant::input[1] 則匹配3個,因為// descendant 從當前節點中任意一級子節點進行定位。

 

對於following::元素節點[num]  這種形式的定位,總結如下

1)、如果定位的元素節點不在子節點內,那么/和// 使用沒有區別。

2)、如果定位的元素節點只在子節點內,那么使用following時只能使用//,但會出現定位到多個元素。 建議使用 / descendant。(//descendant 在任意一級子節點進行定位)

3)、如果定位的元素節點既在子節點內,又在節點外,如果你想定位的是子節點內的,則用 /descendant ; 如果想定位的是節點外的,使用/following,不能使用//following,因為子節點內的元素節點存在,會出現定位到多個元素。

6、Xpath 的常見函數

last()  最后一個節點數

.//*[@id='select2-drop']/ul/li[last()]      last()是代表最后一個li。可以運算last()-3

 count(node-set)    返回節點集node-set中的節點數

 

 可以用於統計

7、頁面滾動條滑動

self.driver.execute_script("window.scrollTo(0,document.body.scrollHeight);")  # 將頁面滾動條滑到底部
self.driver.execute_script("arguments[0].scrollIntoView();", el)  # 向下滑動滾動條,跳轉到目標元素處
self.driver.execute_script("arguments[0].scrollIntoView(false);", el)  # 向上滑動滾動條,跳轉到目標元素處

arguments[0].scrollIntoView(),不能隨意使用,會先把元素element對象的表格“頂端”移動到與當前窗口的“頂部”對齊,如果元素當前可見,可能移動后就不可見了,導致定位報錯。

因為會把元素頂端對齊窗口頂部,有時候也會出現跳轉后,元素仍然不可見的情況。如果是需要點擊這個元素,可以使用下面這種方式:
self.driver.execute_script("arguments[0].click();", el)  # 直接點擊不可見的目標元素,不再先跳轉。

8、獲取某節點下子節點個數

 

 

 

獲取div節點下li的節點個數

法一:num=self.driver.execute_script("return $('.jspPane').find('li').length;") # JQuery寫法 , 要加上return 才能獲取js執行的值!!!

法二:num=self.driver.execute_script('return document.getElementsByClassName("jspContainer")[0].getElementsByTagName("li").length;') #DOM寫法

 

 9、selenium 獲取元素文本值

 self.drive.find_element(*loc).text   文本值加載需要時間,獲取文本值前 常常需要sleep一下 否則就可能獲取為空值 或者 獲取的不是最新的文本值 ,有沒有更好的方法呢?

 

 10、使用find_element() ,如果頁面上有符合條件的多個元素時,find_element 只會找到符合條件的第一個元素! 不會報錯。

11、input輸入框

這是一個input型輸入框,當我點擊或輸入值時,輸入框下方會顯示選擇項幫助快速輸入,代碼如下:

<input class="v-input some" autocomplete="off" name="some"/>

< input  name="any" style="display: none;"/>
< ul  class="V-ss bb ">
< li  class="v-select-item">蘋果</ li >
< li  class="v-select-item">櫻桃</ li >
< li  class="v-select-item">藍莓</ li >
</ ul >

一開始我只定位了inout輸入框,向其中輸入數據“蘋果”,然后點擊查找,這時問題出現了,雖然頁面刷新了,但是並未進行查找

問題存在的原因是,輸入框雖然提供了輸入,但是必須從出現的選擇項中進行點擊,完成選擇,才可進行查找,由此我就需要定位到代碼中相應的li

解決方法1:

首先向輸入框中輸入值,然后下方選項就會只顯示<li>蘋果</li>,然后在查找input輸入框的兄弟元素(following-sibling)下的li(此時是唯一的)

解決方法2:

點擊輸入框,將會顯示下方選項,然后查找input輸入框下兄弟元素下的li(此時是一個列表),然后傳入值,遍歷lil列表,找到和傳入值相同的li的text值,進行點擊

drive.find_element_by_name("some").click()

lilist = drive.find_elements_by_xpath( "//input[@name='some']/following-sibling::ul/li" )
for  li  in  lilist:
     if  li.text = = "蘋果"
         li.click()
12 、無法識別谷歌自帶的彈出框
最近想從某培訓機構網站上爬取題庫,遇到了彈出框無法識別的問題,控制台提示“selenium.common.exceptions.NoAlertPresentException: Message: no such alert”。

 

 然而彈出框是確實存在的啊,設置了元素等待,在報錯之前彈出框已經出現了的。手動點擊沒有問題。

 

 

既然不是元素等待問題,接下來考慮是不是frame不同導致元素無法識別。因為某些彈出框,本身不一定是alert、confirm等類型,也可能是一個彈出頁面(雖然這個彈出框看着不像,還是要排查下)。使用Chrome自帶的開發者工具(F12)發現無法識別該彈出框的任何元素。

開始懷疑這個是Chrome自身的彈出框,而不是網站的彈出框。

驗證:使用Firefox瀏覽器操作,發現不會出現該彈出框,可以確認這是Chrome自身的彈出框。

這種提示一般跟Chrome的設置有關,然而在設置頁面沒有找到相關的配置項。

 

 

這個時候當然是找百度幫忙了,然而在我變換了多個搜索關鍵字后也搜索不到我想要的結果。我就納悶了,難道其他人都沒遇到過這種情況嗎?不過也不奇怪,我也是第一次遇到這種權限申請,一般都是“網站想要發送通知”的那種。

其實換Firefox瀏覽器就不耽誤時間了,我的最終目的是拿到題庫,過程無關緊要。不過我強迫症發作了,這個問題解決后下次遇到同樣的問題就能快速解決了,而且也可以給別人參考。

好吧,我繼續查。這個時候針對性地去查Chrome設置、Chrome權限等。最終查到了解決方法。有時候用英文關鍵字搜索可能更有效……

解決方案:在打開瀏覽器的時候設置--unlimited-storage,然后刪掉報錯的代碼driver.switch_to.alert.accept()。這樣就不會彈出提示框,而且網站也能正常展示了。

options=webdriver.ChromeOptions()
options.add_argument('--unlimited-storage')
driver=webdriver.Chrome(chrome_options = options)


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM