java - Selenium ChromeDriver:增加获取 WebElement 文本的时间
问题描述
我有一个代码,我在其中遍历表的行和列,我想将它的值添加到列表中。
这需要我大量的时间。
所以我添加了一个时间测量,我注意到由于某种原因,时间逐行增加。
我不明白为什么。
你能建议吗?
private void buildTableDataMap() {
WebElement table = chromeWebDriver.findElement(By.id("table-type-1"));
List<WebElement> rows = table.findElements(By.tagName("tr"));
theMap.getInstance().clear();
String item;
for (WebElement row : rows) {
ArrayList<String> values = new ArrayList<>();
List<WebElement> tds = row.findElements(By.tagName("td"));
if(tds.size() > 0){
WebElement last = tds.get(tds.size() - 1);
long time = System.currentTimeMillis();
values.addAll(tds.stream().map(e->e.getText()).collect(Collectors.toList()));
System.out.println(System.currentTimeMillis() - time);
//remove redundant last entry:
values.remove(tds.size() - 1);
callSomeFunc(values, last);
item = tds.get(TABLE_COLUMNS.NAME_COL.getNumVal()).getText();
item = item.replaceAll("[^.\\- /'&A-Za-z0-9]", "").trim();//remove redundant chars
theMap.getInstance().getMap().put(item, values);
}
}
}
伙计们,我继续研究。首先,弗洛伦特的友好回答对我没有帮助,因为至少据我了解,它返回了我必须解析的字符串数组列表,而且我不太喜欢这种解决方案......
因此,我发现e.getText()调用从调用到调用的时间都在增加,从而解决了问题!我也尝试了 e.getAttribute("innerText")但没有改变。无法理解为什么。有什么想法可以解决吗?
WebElement last = null;
for (WebElement e : tds){
last = e;
long tm1 = 0, tm2 = 0;
if(Settings.verboseYN) {
tm1 = System.currentTimeMillis();
}
s = e.getText(); //This action increases in time!!!
if(Settings.verboseYN) {
tm2 = System.currentTimeMillis();
}
values.add(s); //a 0 ms action!!!
if(Settings.verboseYN) {
System.out.println("e.getText()) took " + (tm2 - tm1) + " ms...");
}
}
这是 getText 所用时间的图表......
2018 年 5 月 8 日另一个增加执行时间的来源是:
void func(WebElement anchorsElement){
List<WebElement> anchors = anchorsElement.findElements(By.tagName("a"));
for (WebElement a : anchors) {
if (a.getAttribute("class").indexOf("a") > 0)
values.add("A");
else if (a.getAttribute("class").indexOf("b") > 0)
values.add("B");
else if (a.getAttribute("class").indexOf("c") > 0)
values.add("C");
}
}
每个函数只有 5 次迭代,但每次调用该函数仍会增加其执行时间。这个也有解决方案吗?
解决方案
呼叫驱动程序是一项昂贵的操作。要显着减少执行时间,请使用 JavaScript 注入executeScript
在一次调用中读取整个表。然后使用 Java 在客户端处理/过滤数据。
public ArrayList<?> readTable(WebElement table)
{
final String JS_READ_CELLS =
"var table = arguments[0]; " +
"return map(table.querySelectorAll('tr'), readRow); " +
"function readRow(row) { return map(row.querySelectorAll('td'), readCell) }; " +
"function readCell(cell) { return cell.innerText }; " +
"function map(items, fn) { return Array.prototype.map.call(items, fn) }; " ;
WebDriver driver = ((RemoteWebElement)table).getWrappedDriver();
Object result = ((JavascriptExecutor)driver).executeScript(JS_READ_CELLS, table);
return (ArrayList<?>)result;
}
推荐阅读
- find - 删除找到特定文件的目录中的所有文件
- python - 为什么 "9999999999999999 / 16 = 625000000000000.0" 而 "9999999999999999 // 16 = 624999999999999"
- reactjs - NX 和 NextJS monorepo 中的全局类型声明
- inno-setup - 在 Inno Setup 的许可证页面上添加一个打印按钮(针对 Inno Setup 6 重新访问)
- javascript - scalajs-react:如何从状态对象的 Seq 中获得多行文本字段?
- python - 如何在python中输入用户的字典列表?
- python - Python OSError:[Errno 22] 将 pd.read_csv 与两个 csv 文件一起使用时参数无效
- azure - 如何使用 kafka 使用来自 azure eventthub 的审核消息?
- python - Python、Zingchart
- reactjs - 如何使用 React.js 在 MongoDB Realm Web APP 上设置当前项目