使用 Chrome 时硒 "selenium.common.exceptions.NoSuchElementException"

Selenium "selenium.common.exceptions.NoSuchElementException" when using Chrome

我正在尝试在 Chrome 上使用 Selenium 玩 QWOP,但我不断收到以下错误:

selenium.common.exceptions.NoSuchElementException: 
Message: no such element: Unable to locate element
{"method":"id","selector":"window1"
(Session info: chrome=63.0.3239.108
(Driver info: chromedriver=2.34.522913
(36222509aa6e819815938cbf2709b4849735537c), platform=Linux 4.10.0-42-generic x86_64)

同时使用以下代码:

from selenium import webdriver
from selenium.webdriver.common.action_chains import ActionChains
import time

browser = webdriver.Chrome()
browser.set_window_size(640, 480)
browser.get('http://www.foddy.net/Athletics.html?webgl=true')
browser.implicitly_wait(10)

canvas = browser.find_element_by_id("window1")

canvas.click()

while (True):
    action = ActionChains(browser)
    action.move_to_element(canvas).perform()
    canvas.click()
    canvas.send_keys("q")

相同的代码在 Firefox 上运行完美,但因为我想使用 chrome 的功能来 运行 无头模式下的 webgl 游戏,我不能真正切换到 Firefox。

有任何变通方法可以使它正常工作吗?

NoSuchElementException

selenium.common.exceptions.NoSuchElementException俗称NoSuchElementException定义为:

exception selenium.common.exceptions.NoSuchElementException(msg=None, screen=None, stacktrace=None)

NoSuchElementException基本上是2种情况抛出如下:

  • 使用时:

    webdriver.find_element_by_*("expression")
    //example : my_element = driver.find_element_by_xpath("xpath_expression")
    
  • 使用时:

    element.find_element_by_*("expression")
    //example : my_element = element.find_element_by_*("expression")
    

根据 API 文档,就像任何其他文档一样 selenium.common.exceptionsNoSuchElementException 应该包含以下参数:

  • 消息、屏幕、堆栈跟踪

        raise exception_class(message, screen, stacktrace)
    selenium.common.exceptions.NoSuchElementException: Message: no such element: Unable to locate element: {"method":"xpath","selector":".//*[@id='create-portal-popup']/div[4]/div[1]/button[3]"}
      (Session info: chrome=61.0.3163.100)
      (Driver info: chromedriver=2.32.498550 (9dec58e66c31bcc53a9ce3c7226f0c1c5810906a),platform=Windows NT 10.0.10240 x86_64)
    

原因

NoSuchElementException 的原因可以是以下之一:

  • 您采用的定位器策略没有识别HTML DOM中的任何元素。
  • 您采用的定位器策略无法识别该元素,因为它不在浏览器的Viewport.
  • 您采用的 Locator 策略 标识了元素,但由于属性 style="display: none;" 的存在而不可见.
  • 您采用的定位器策略并未唯一地识别[=219=中的所需元素] DOM 并且目前发现了其他一些 隐藏 / 不可见 元素。
  • 您要查找的 WebElement 位于 <iframe> 标记内。
  • WebDriver 实例正在查找 WebElement 甚至在 [=99= 中的元素 present/visibile 之前]HTMLDOM.

解决方案

解决NoSuchElementException的解决方案可以是以下任意一种:

  • 采用一个 来唯一标识所需的 WebElement。您可以借助 Developer Tools (Ctrl+Shift+IF12) 并使用 Element Inspector

    这里有关于how to inspect element in selenium3.6 as firebug is not an option any more for FF 56?

  • 的详细讨论
  • 使用execute_script()方法将元素滚动进去查看如下:

    elem = driver.find_element_by_xpath("element_xpath")
    driver.execute_script("arguments[0].scrollIntoView();", elem)
    

    这里有关于

  • 的详细讨论
  • Incase 元素具有属性 style="display: none;",通过 executeScript() 方法删除属性如下:

    elem = driver.find_element_by_xpath("element_xpath")
    driver.execute_script("arguments[0].removeAttribute('style')", elem)
    elem.send_keys("text_to_send")
    
  • 检查元素是否在 <iframe> 内遍历 HTML 以定位相应的 <iframe>通过以下任一方法标记和 switchTo() 所需的 iframe :

    driver.switch_to.frame("iframe_name")
    driver.switch_to.frame("iframe_id")
    driver.switch_to.frame(1) // 1 represents frame index
    

    在这里你可以找到关于的详细讨论。

  • 如果元素在HTML中不存在/可见 DOM 立即将 WebDriverWait with expected_conditions 设置为正确的方法如下:

    • 等待presence_of_element_located:

      element = WebDriverWait(driver, 20).until(expected_conditions.presence_of_element_located((By.XPATH, "element_xpath']")))
      
    • 等待visibility_of_element_located:

      element = WebDriverWait(driver, 20).until(expected_conditions.visibility_of_element_located((By.CSS_SELECTOR, "element_css")
      
    • 等待element_to_be_clickable:

      element = WebDriverWait(driver, 20).until(expected_conditions.element_to_be_clickable((By.LINK_TEXT, "element_link_text")))
      

这个用例

您看到的是 NoSuchElementException,因为 id 定位器没有唯一标识 canvas。要识别上面的 canvas 和 click(),您必须等待 canvas 变为 clickable 并实现它,您可以使用以下代码块:

WebDriverWait(driver, 20).until(EC.element_to_be_clickable((By.XPATH, "//canvas[@id='window1']"))).click()

参考

您可以在以下位置找到 's 基于客户的相关讨论: