Selenium无法定位元素的解决方案合集

2023-02-18  本文已影响0人  DD丿

WebDriver只能在一个页面上对元素识别与定位,对于frame/iframe表单内嵌的页面元素无法直接定位。

解决方法

switch_to.frame()默认可以直接取表单的id或name属性。如果没有可用的id和name属性,可以先定位到frame/iframe,再将定位对象传给switch_to.frame(对象)方法。
xf = driver.find_element_by_xpath('//*[@class="if"]') driver.switch_to.frame(xf) ...
driver.switch_to.parent_frame()切到父frame。影响性能,可以提给开发,让其改进。 driver.switch_to.default_content() 跳回最外层的页面

02

页面跳转到新标签页,或弹出警告框等

在页面操作过程中有时候点击某个链接会弹出新窗口,这时就需要切换焦点到新窗口上进行操作。

alert的方法有:

03

页面元素失去焦点导致脚本运行不稳定

解决方法

driver.switch_to.active_element 遇到脚本不稳定,有时会失去焦点导致测试失败的情况下,可以先切到焦点元素再进行操作。注意.active_element后面不带括号()。

下面是一个参考案例:

'最初的 “右击鼠标 → 新建文件夹 → 输入文件夹名称” 的代码' l = driver.find_element_by_id('pm_treeRoom_1_span') ActionChains(driver).context_click(l).perform() driver.find_element_by_class_name('fnew').click() time.sleep(2) driver.find_element_by_xpath('//*[@id="pm_treeRoom_1_ul"]/li[...]').send_keys('filename') time.sleep(2)

结果这种操作总会导致输入框失去焦点,直接消失,更不能send_keys进去了,直接报错。

'修改后的代码如下' driver.find_element_by_class_name('fnew').click() time.sleep(2) driver.switch_to.active_element.send_keys('filename') time.sleep(2)

04

使用Xpath或CSS定位

find_element_by_xpath("//标签[属性='值']")

使用Xpath/CSS方法,非常适合定位属性值动态生成、不容易定位的元素。如果不想指定标签,则可以使用“*”代替,使用xpath不局限于id、name和class这三个属性,元素的任意属性值都可以使用,只要它能唯一的标识一个元素。

05

页面还没加载出来就对页面上的元素进行操作

因为加载元素延时造成的脚本失败,我们可以通过设置等待时间来提升自动化脚本的稳定性。

解决方法1

WebDriverWait()显示等待。等待单个的元素加载,通常配合until()until_not()方法使用。

即,WebDriverWait(driver, 超时时长, 调用频率, 忽略异常).until(可执行方法, 超时时返回的信息)

最长等待时间为5s,每隔1秒检查一次id='kw'的元素是否被加载在DOM树里(并不代表该元素一定可见)。最常用的method是expected_conditions类提供的预期条件判断。

最长等待时间为30s,每隔1秒检查一次id='someId'的元素是否从DOM树里消失,忽略默认异常信息NoSuchElementException 和指定的异常信息ElementNotVisibleException。此处匿名函数lambda的用法具体参考Python语法。

解决方法2

driver.implicitly_wait(秒) 隐式等待。全局等待,对所有元素设置超时时间,等待页面的加载,因此只需要设置一次即可。这里的时间是最长等待时间(非固定等待时间)。

解决方法3

sleep(秒)线程等待。休眠固定的时间,使用时需要先引入time模块的sleep方法from time import sleep。

06

元素被遮挡,不可用,不可见

07

用WebDriver调用JavaScript代码代替无法实现的功能

对于有些WebDriver没有提供的方法或者无法实现的功能,WebDriver提供了driver.execute_script()方法来执行JavaScript代码。

解决方法

如果页面内容过长,窗口最大化也无法查看到所有元素,可以通过执行JavaScript脚本实现滚动条的拖动等动作。
以上语句实现了拉动页面到底部的功能,其中window.scrollTo(左边距,上边距)是JavaScript中用于设置浏览器窗口滚动条的水平和垂直位置的代码。

text = "input text" driver.execute_script("var obj=document.getElementById('text'); obj.value=' " + text + " ';")

假设一个输入框可以通过id='text'将其定位,却不能通过send_keys()输入文本内容,可以借助JavaScript代码来实现。

video = driver.find_element_by_xpath("body/Section[1]/div/video") url = driver.execute_script("return arguments[0].currentSrc;", video) print(url) '返回文件播放地址' print("start") '播放视屏' driver.execute_script("return arguments[0].play()", video) sleep(15) '播放15秒钟' print(stop) '暂停视屏' driver.execute_script("arguments[0].pause()", video) ...

以上实现了HTML5视屏<video>标签的部分测试,更多内容参考HTML DOM Video对象。

其中arguments是JavaScript的内置对象。因为将video对象传给了arguments,所以arguments[0]相当于JavaScript脚本的document.getElementsByTagName("video")。JavaScript不支持重载,使用arguments对象可以模拟函数重载效果。

08

WebDriver无法操作Windows控件

文件的普通上传和下载(参考How to auto save files using custom Firefox profile ?),可以通过.``.send_keys('本地路径')和``find_element_by_partial_link_text('下载链接名').click()实现。

解决方法

对于插件上传,需要操作Windows控件的,可以通过安装AutoIt工具、编写脚本、保存为“.au3”文件、转换成“.exe”文件,再由自动化脚本os.system("D:\\upfile.exe")实现上传/下载。

  • 虽然这种方法可以解决文件上传、下载的操作问题,但是并不推荐。因为通过python调用exe程序并不在python的可控范围内,执行多长时间,执行过程是否出错,都无从自动化过程得知。PS:传图片/程序/视频类直接调用pyaotugui库pyautogui.write(r'C:\cover.png')

09

firefox安全性强,不允许跨域调用出现报错

错误描述:

解决办法

Firefox 要取消XMLHttpRequest的跨域限制的话,

try {

netscape.security.PrivilegeManager.enablePrivilege("UniversalBrowserRead"); } catch (e) { alert("Permission UniversalBrowserRead denied."); }

上一篇 下一篇

猜你喜欢

热点阅读