尝试单击多个webelement时获取org.openqa.selenium.StaleElementReferenceException

时间:2015-07-25 12:22:00

标签: exception selenium selenium-webdriver webdriver

我正在尝试测试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

点击第一个产品/元素后,我得到了这个例外 -

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

但我又得到了同样的例外。

然后我放了一个explicitwait而不是隐式等待,但又得到了相同的异常。

然后我继续;在try块以及catch块中但得到了相同的异常。

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

在chrome上我得到了StaleElementReferenceException但是对于firefox我没有在第一次点击后得到任何异常。首次点击后,Firefox浏览器会冻结。

4 个答案:

答案 0 :(得分:1)

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

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

  • 将元素提取到集合
  • 与其中一个进行互动,将您带到上一页
  • 导航回来
  • 尝试重复上述内容,同时已经发生了2页重新加载

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

您需要做的是:

  • 创建ProductListPage
  • 实现int getNumberOfProducts(),它将返回页面上显示的产品数量
  • 实现void clickProduct(int index),它将带您进入产品详细信息页面
  • 创建ProductDetailsPage
  • 实施方法退回

在你的测试类中,你应该循环到getNumberOfProducts()。在这个循环中,你可以:

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

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

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

答案 1 :(得分:1)

发生了

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。请点击此链接以便更好地理解; http://docs.seleniumhq.org/docs/04_webdriver_advanced.jsp#explicit-and-implicit-waits

答案 2 :(得分:1)

  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()方法中。到目前为止,代码对我来说完全正常。

答案 3 :(得分:0)

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

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

控制台输出 enter image description here