尝试点击多个网络元素时获得 org.openqa.selenium.StaleElementReferenceException

Getting org.openqa.selenium.StaleElementReferenceException when try to click on multiple webelement

我正在尝试测试这个 Website。它里面有多种产品。我想点击第一个产品,打开产品页面后我想返回上一页,然后点击第二个产品。打开后,我想再次返回,然后对网页上提供的所有产品执行相同的操作。

我写了下面的代码来做到这一点 -

  driver.manage().timeouts().implicitlyWait(8000, TimeUnit.SECONDS);
   List<WebElement> product = driver.findElements(By.xpath(".//*[@class='products-grid products-grid--max-4-col first last odd']/li/a"));
   for(int i=0; i<product.size();i++){
       try{
       driver.manage().timeouts().implicitlyWait(8000, TimeUnit.SECONDS);
       product.get(i).click();
       System.out.println(i);
       System.out.println("Title is : "+driver.getTitle());
       System.out.println("Product URL is : "+driver.getCurrentUrl());
       driver.navigate().back();
       }catch(StaleElementReferenceException e){
           e.printStackTrace();
       }//catch
   }//for

点击第一个 product/element 后出现异常 -

org.openqa.selenium.StaleElementReferenceException: stale element reference:  element is not attached to the page document
(Session info: chrome=44.0.2403.89)
(Driver info: chromedriver=2.9.248315,platform=Windows NT 6.1 SP1 x86)(WARNING: The server did not provide any stacktrace information)
Command duration or timeout: 115 milliseconds
For documentation on this error, please visit: http://seleniumhq.org/exceptions/stale_element_reference.html
Build info: version: '2.46.0', revision: '87c69e2', time: '2015-06-04 16:17:10'
System info: host: 'ShantanuNandan', ip: '10.0.0.4', os.name: 'Windows 7', os.arch: 'x86', os.version: '6.1', java.version: '1.8.0_45'
Driver info: org.openqa.selenium.chrome.ChromeDriver
Capabilities [{applicationCacheEnabled=false, rotatable=false, chrome= {userDataDir=C:\Users\SHANTA~1\AppData\Local\Temp\scoped_dir6172_17060},  takesHeapSnapshot=true, databaseEnabled=false, handlesAlerts=true,  version=44.0.2403.89, platform=XP, browserConnectionEnabled=false,  nativeEvents=true, acceptSslCerts=true, locationContextEnabled=true,  webStorageEnabled=true, browserName=chrome, takesScreenshot=true,  javascriptEnabled=true, cssSelectorsEnabled=true}]
Session ID: 081ee44a67affb195766d4fc8ce165d3
at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
at sun.reflect.NativeConstructorAccessorImpl.newInstance(Unknown Source)
at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(Unknown Source)
at java.lang.reflect.Constructor.newInstance(Unknown Source)
at  org.openqa.selenium.remote.ErrorHandler.createThrowable(ErrorHandler.java:204)
at  org.openqa.selenium.remote.ErrorHandler.throwIfResponseFailed(ErrorHandler.java:156)
at  org.openqa.selenium.remote.RemoteWebDriver.execute(RemoteWebDriver.java:605)
at  org.openqa.selenium.remote.RemoteWebElement.execute(RemoteWebElement.java:269)
at  org.openqa.selenium.remote.RemoteWebElement.click(RemoteWebElement.java:80)
at  com.Selenium_Practice.Niraame_Login_Logout_ElementDisplay.ClickAndComeBack(Niraame_Login_Logout_ElementDisplay.java:80)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)
at java.lang.reflect.Method.invoke(Unknown Source)
at  org.testng.internal.MethodInvocationHelper.invokeMethod(MethodInvocationHelper.java:84)
at org.testng.internal.Invoker.invokeMethod(Invoker.java:714)
at org.testng.internal.Invoker.invokeTestMethod(Invoker.java:901)
at org.testng.internal.Invoker.invokeTestMethods(Invoker.java:1231)
at  org.testng.internal.TestMethodWorker.invokeTestMethods(TestMethodWorker.java:127)
at org.testng.internal.TestMethodWorker.run(TestMethodWorker.java:111)
at org.testng.TestRunner.privateRun(TestRunner.java:767)
at org.testng.TestRunner.run(TestRunner.java:617)
at org.testng.SuiteRunner.runTest(SuiteRunner.java:334)
at org.testng.SuiteRunner.runSequentially(SuiteRunner.java:329)
at org.testng.SuiteRunner.privateRun(SuiteRunner.java:291)
at org.testng.SuiteRunner.run(SuiteRunner.java:240)
at org.testng.SuiteRunnerWorker.runSuite(SuiteRunnerWorker.java:52)
at org.testng.SuiteRunnerWorker.run(SuiteRunnerWorker.java:86)
at org.testng.TestNG.runSuitesSequentially(TestNG.java:1224)
at org.testng.TestNG.runSuitesLocally(TestNG.java:1149)
at org.testng.TestNG.run(TestNG.java:1057)
at org.testng.remote.RemoteTestNG.run(RemoteTestNG.java:111)
at org.testng.remote.RemoteTestNG.initAndRun(RemoteTestNG.java:204)
at org.testng.remote.RemoteTestNG.main(RemoteTestNG.java:175) 

我在论坛上遇到了同样的问题,到目前为止做了这些事情。 首先,我放置 Thread.sleep 并刷新页面以使该元素可用于单击。 代码-

driver.navigate().back();
Thread.sleep(4000);
driver.navigate().refresh();

但我又遇到了同样的异常。

然后我设置了显式等待而不是隐式等待,但再次遇到相同的异常。

那我继续;在 try 块和 catch 块中,但出现相同的异常。

我又尝试了2个选项,但都没有用。 请告诉我我在哪里做错了。 我知道它为什么会发生(它会发生是因为 DOM 在第一次点击后没有使元素可用)但是我无法找到一种方法来获得我想要的结果。

在 chrome 我得到 StaleElementReferenceException 但对于 firefox 我在第一次点击后没有得到任何异常。第一次点击后 firefox 浏览器卡住。

Selenium 在此交互之前将内部元素定位器分配给您要与之交互的每个元素。您可以在执行日志中跟踪此行为。在您的示例中,这些定位器是在您执行 driver.findElements.

时分配的

如果在 a) 分配这些定位器和 b) 您的代码与这些元素之间的交互之间刷新页面或 DOM 模型发生变化,将抛出 StaleElementReferenceException。 这就是您的代码中发生的情况。你:

  • 获取元素到集合
  • 与其中之一互动,这会将您带到上一页
  • 返回
  • 尝试重复以上操作,同时已重新加载 2 次页面

您的问题的答案称为页面对象设计模式,例如here.

您特别需要做的是:

  • 创建产品列表页面
  • 实施 int getNumberOfProducts(),这将 return 页面上显示的产品数量
  • 执行 void clickProduct(int index),这将带您进入产品详情页面
  • 创建产品详细信息页面
  • 后退的实现方法

在您的测试 class 中,您应该执行 for 循环直到 getNumberOfProducts()。在这个循环中,你做:

  • productListPage.clickProduct(i)
  • 等待显示 ProductDetailsPage (x)
  • 返回上一页
  • 等待显示 ProductListPage (x)

(x) - 您可以为这 2 个页面对象实现自定义 waitForPageToLoad() 方法,这将等待直到显示该特定页面的唯一元素。

我在这里看到的可能问题是产品详细信息页面上缺少 return link,因此您使用浏览器后退按钮应用了解决方法。如果您必须使用 SafariDriver 自动化它,这将是一个问题,其中未实现与浏览器后退和前进按钮的交互。

发生

StaleElementRefreshException 是因为当您导航回上一页时,DOM 正在刷新,而 List 元素正在变得陈旧。

您可以再次在 for 循环中添加 创建列表(您之前添加过) 的代码。因此,每当页面导航回前一个页面时,列表就会再次刷新。

请看下面更新的代码:

driver.manage().timeouts().implicitlyWait(8, TimeUnit.SECONDS);
List<WebElement> product = driver.findElements(By.xpath(".//*[@class='products-grid products-grid--max-4-col first last odd']/li/a"));
for(int i=0; i<product.size();i++){
   try{
       //For Refreshing the list each time page gets navigated back.
       product = driver.findElements(By.xpath(".//*[@class='products-grid products-grid--max-4-col first last odd']/li/a"));

       product.get(i).click();
       System.out.println(i);
       System.out.println("Title is : "+driver.getTitle());
       System.out.println("Product URL is : "+driver.getCurrentUrl());
       driver.navigate().back();
       }catch(StaleElementReferenceException e){
           e.printStackTrace();
       }//catch
   }//for

很少有建议:

  1. 请避免使用那么长的隐式等待时间。这是 8000 秒,大约等于 2 小时 22 分钟。显然等待一个元素太多了.. :)
  2. 如果要使用隐式等待,一开始就使用。一旦设置,每次驱动程序尝试查找元素时都会应用等待时间。无需每次都添加。而且,如果你想等待超过隐式给定的时间,你可以使用显式等待相关的 WebElement。请遵循此 link 以获得更好的理解; http://docs.seleniumhq.org/docs/04_webdriver_advanced.jsp#explicit-and-implicit-waits

昨天我也在尝试这个问题。它对我有用。这是工作代码。

import java.util.List;
import java.util.concurrent.TimeUnit;
import org.openqa.selenium.By;
import org.openqa.selenium.WebDriver;
import org.openqa.selenium.WebElement;
import org.openqa.selenium.firefox.FirefoxDriver;
import org.openqa.selenium.remote.service.DriverService;

public class clickOnAllTabOfPage {

    public static void main(String[] args) {
        WebDriver driver = new FirefoxDriver();
        driver.manage().timeouts().implicitlyWait(10, TimeUnit.SECONDS);
        driver.manage().window().maximize();
        driver.get("http://niraame.com/beauty.html");   
        List <WebElement> Items = driver.findElements(By.xpath("//*[@class='products-grid products-grid--max-4-col first last odd']/li/a"));
        int lenght=Items.size();
        for (int i=0; i<lenght; i++) {      
            try {
                Items.get(i).click();
                System.out.println("i value :"+i+ "|title is :" +driver.getTitle() + "|Item Url is :"+driver.getCurrentUrl());
                driver.navigate().back();
                Items = driver.findElements(By.xpath("//*[@class='products-grid products-grid--max-4-col first last odd']/li/a"));

            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        System.out.println("Opened all the links on the Page and hence closing the browser");
        driver.quit();
    }
}

控制台输出

  public void ClickAndComeBack() throws InterruptedException, IOException{
   driver.manage().timeouts().implicitlyWait(8000, TimeUnit.SECONDS);
   List<WebElement> productLink = driver.findElements(By.xpath(".//*[@class='products-grid products-grid--max-4-col first last odd']/li/a"));
   String[] str = new String[productLink.size()];
   for(int i=0; i<productLink.size();i++){
       str[i]=productLink.get(i).getAttribute("href");
   }//for
   int i=0;
   for(String strr: str){
       driver.manage().timeouts().implicitlyWait(8000, TimeUnit.SECONDS);
       driver.get(strr);
       System.out.println("Title is : "+driver.getTitle());
       System.out.println("Product URL is : "+driver.getCurrentUrl());
       File src = ((TakesScreenshot)driver).getScreenshotAs(OutputType.FILE);
       FileUtils.copyFile(src, new File("D:\NiraameProductScreenShot\pic"+i+".jpeg"), true);
       i++;
       driver.navigate().back();
       Thread.sleep(4000);
   }//for
 }//ClickAndComeBack

在第一个 for 循环中,我将所有 href 存储在一个字符串数组中。在第二个循环中,我将存储的数组放入 webdriver 的 driver.get() 方法中。到目前为止,代码对我来说工作得很好。