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()

然而彈出框是確實存在的啊,設置了元素等待,在報錯之前彈出框已經出現了的。手動點擊沒有問題。
既然不是元素等待問題,接下來考慮是不是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)