【问题标题】:Selenium StaleElementReferenceExceptionSelenium StaleElementReferenceException
【发布时间】:2017-01-22 14:50:11
【问题描述】:

您好,我是Selenium的新手

我正在使用Java library,已经尝试过ChromeFirefox 驱动程序。

我正在运行一个循环。有趣的是,循环有时会工作 3、2 次,它并不总是在同一次迭代中失败。我认为它与某种竞争条件有关(例如等待页面加载)。如果我在调试模式下运行,它似乎工作得很好。

我已经尝试过来自其他答案的建议,例如 wait explicitlyimplicitly,但仍然没有帮助。也许如果你看到代码,你可以帮我一把。

这进入一个循环。

WebDriverWait wait = new WebDriverWait(driver,20);
WebElement searchResults = driver.findElement(new By.ById("searchresults"));
searchResults = searchResults.findElement(new By.ByClassName("table"));
wait.until(ExpectedConditions.presenceOfAllElementsLocatedBy(By.tagName("a")));
List<WebElement> list=searchResults.findElements(By.tagName("a"));
for(WebElement w: list) {
  result.add(w.getAttribute("href")); //EXCEPTION HAPPENS ALWAYS HERE
}

解决方案

解决方案是彻底破解。我仍然不明白,但它的工作。如果有人明白为什么,请告诉我。

我只是将所有的等待时间提高,它的性能会更好。我还接受了@Cyril 的建议,如果在一些数据检查中抛出异常以确保我得到了我想要的一切,则重新运行迭代。

WebDriverWait wait = new WebDriverWait(driver,20);
wait.until(ExpectedConditions.presenceOfAllElementsLocatedBy(new By.ById("searchresults")));
wait.until(ExpectedConditions.presenceOfAllElementsLocatedBy(new By.ByClassName("table"))); 
wait.until(ExpectedConditions.presenceOfAllElementsLocatedBy(By.tagName("a")));

【问题讨论】:

    标签: java selenium web-crawler


    【解决方案1】:

    StaleElementReferenceException 表示您正在访问的 DOM 元素已更改或从 DOM 中删除。所以等到元素可见或存在并没有帮助(之后它们仍然可以更新)。

    在查找任何 DOM 元素之前,您可以通过 Thread.sleep 显式等待 5 秒来修复异常。

    另一个选项是look up the element again if the exception is thrown。但它只适用于单个元素。

    【讨论】:

    • 非常感谢您的回答我采纳了您的最后建议。我以某种方式对其进行了一些其他更改(见上文)。我不是 100% 开心,但它就是这份工作。
    【解决方案2】:

    为了避免StaleElementReferenceException,您可以在查找元素期间获取该属性。这是一个 sn-p(nth-child 第一个索引是 1):

    List<WebElement> list=searchResults.findElements(By.tagName("a"));
    for(int i = 1; i < list.length; i++) {
       result.add(searchResults.findElement(By.cssSelector("a:nth-child(" + i + ")")).getAttribute("href"));
    }
    

    这样,例外的唯一机会是元素在 findElementgetAttribute 之间发生变化,这是不太可能的(或者表格元素发生变化 - searchResults)。 当然,有更好的方法来实现这一点,比如围绕findElement 的包装函数或使用页面对象模式,但对于您的情况,它应该就足够了。

    编辑:您的解决方案可能有效,因为它使脚本等到页面完全加载,然后元素不再更改。

    【讨论】:

    • @Altober,看看这个解决方案,看起来很完美。
    • 嗨,非常感谢。我试过了,但由于某种原因,经过几次迭代,执行挂在 for 循环中。我会继续探索并带来一些更新。
    • 感谢@Grzegorz 让我开心;)。至于挂起,我认为这是因为您有很多来自 a 标记的元素,并且它会为每个元素轮询 DOM ...因此,如果您可以通过使用更具体的定位器来缩小列表,我假设它会有所帮助。即By.cssSelector("a.someClass")
    猜你喜欢
    • 2011-06-18
    • 2018-09-14
    • 1970-01-01
    • 2016-04-28
    • 2014-11-15
    • 1970-01-01
    • 2015-01-16
    • 2015-02-17
    • 1970-01-01
    相关资源
    最近更新 更多