python+selenium頁面自動化 元素定位實際遇到的各種問題(持續(xù)更新)

https://www.cnblogs.com/who-care/p/7908703.html

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

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

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

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

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

可參考 https://blog.csdn.net/cyjs1988/article/details/75006167?


2、上傳文件

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

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

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


3、文件下載

方法一:(不推薦使用)

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

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

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

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

方法二:(推薦)

通過設(shè)置瀏覽器參數(shù)

option = webdriver.ChromeOptions()

prefs = {'profile.default_content_settings.popups': 0, 'download.default_directory': export_path}? # 設(shè)置下載路徑,關(guān)閉確認(rèn)彈出框。

option.add_experimental_option('prefs', prefs)

driver = webdriver.Chrome(chrome_options=option)

以上是chrome的設(shè)置方法,firefox類似。


4、/.. 定位? 返回上級節(jié)點? 和parent 效果一樣

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

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


5、following 結(jié)合 / 、// 的用法深入分析

following:選取文檔中?當(dāng)前節(jié)點的結(jié)束標(biāo)簽之后?的所有節(jié)點,除屬性節(jié)點(節(jié)點中的屬性)命名空間節(jié)點(類似這樣的節(jié)點

?
??)之外,如果當(dāng)前節(jié)點有子節(jié)點,則包含其所有子節(jié)點。

xxx?/?following:: 選取xxx結(jié)束標(biāo)簽之后的所有節(jié)點,不包括xxx的所有子節(jié)點。? '/' 表示從xxx后進(jìn)行定位,

xxx?//?following:: 選取xxx結(jié)束標(biāo)簽之后的所有節(jié)點,?包括xxx的所有子節(jié)點'//' 表示可以從xxx任意一級子節(jié)點進(jìn)行定位,以及xxx節(jié)點后進(jìn)行定位。


如上圖 想定位到第一個input節(jié)點


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

結(jié)果:沒有匹配元素,

原因:需要定位input屬于當(dāng)前節(jié)點div[@id='History_Table']的子節(jié)點,而/ following 不包括當(dāng)前節(jié)點的子節(jié)點。


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

結(jié)果:匹配到了當(dāng)前節(jié)點中3個input子節(jié)點。

原因:// following? 從當(dāng)前節(jié)點后的任意一級子節(jié)點進(jìn)行定位,所以3個都能匹配到。

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

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


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

?結(jié)果:成功定位到了第一個input

原因:descendant 選取當(dāng)前節(jié)點的所有后代元素,/ descendant 從當(dāng)前節(jié)點后進(jìn)行定位,所以唯一

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


對于following::元素節(jié)點[num]? 這種形式的定位,總結(jié)如下

1)、如果定位的元素節(jié)點不在子節(jié)點內(nèi),那么/和// 使用沒有區(qū)別。

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

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

?

6、Xpath 的常見函數(shù)

last()最后一個節(jié)點數(shù)

.//*[@id='select2-drop']/ul/li[last()]? ? ? last()是代表最后一個li??梢赃\算last()-3

?count(node-set)??返回節(jié)點集node-set中的節(jié)點數(shù)

可以用于統(tǒng)計


7、頁面滾動條滑動

self.driver.execute_script("window.scrollTo(0,document.body.scrollHeight);")? # 將頁面滾動條滑到底部

self.driver.execute_script("arguments[0].scrollIntoView();", el)? # 向下滑動滾動條,跳轉(zhuǎn)到目標(biāo)元素處

self.driver.execute_script("arguments[0].scrollIntoView(false);", el)? # 向上滑動滾動條,跳轉(zhuǎn)到目標(biāo)元素處

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

因為會把元素頂端對齊窗口頂部,有時候也會出現(xiàn)跳轉(zhuǎn)后,元素仍然不可見的情況。如果是需要點擊這個元素,可以使用下面這種方式:

self.driver.execute_script("arguments[0].click();", el)? # 直接點擊不可見的目標(biāo)元素,不再先跳轉(zhuǎn)。


8、獲取某節(jié)點下子節(jié)點個數(shù)


獲取div節(jié)點下li的節(jié)點個數(shù)

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

法二: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 只會找到符合條件的第一個元素! 不會報錯。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容