尝试点击多个网络元素时获得 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
很少有建议:
- 请避免使用那么长的隐式等待时间。这是 8000 秒,大约等于 2 小时 22 分钟。显然等待一个元素太多了.. :)
- 如果要使用隐式等待,一开始就使用。一旦设置,每次驱动程序尝试查找元素时都会应用等待时间。无需每次都添加。而且,如果你想等待超过隐式给定的时间,你可以使用显式等待相关的 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() 方法中。到目前为止,代码对我来说工作得很好。
我正在尝试测试这个 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
很少有建议:
- 请避免使用那么长的隐式等待时间。这是 8000 秒,大约等于 2 小时 22 分钟。显然等待一个元素太多了.. :)
- 如果要使用隐式等待,一开始就使用。一旦设置,每次驱动程序尝试查找元素时都会应用等待时间。无需每次都添加。而且,如果你想等待超过隐式给定的时间,你可以使用显式等待相关的 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() 方法中。到目前为止,代码对我来说工作得很好。