Selenium WebDriver:List <webelement>中的每个WebElement中的findElement()始终返回第一个元素的内容</webelement>

时间:2015-04-02 22:37:48

标签: java selenium selenium-webdriver webdriver

Page page = new Page();
page.populateProductList( driver.findElement( By.xpath("//div[@id='my_25_products']") ) );

class Page
{
    public final static String ALL_PRODUCTS_PATTERN = "//div[@id='all_products']";

    private List<Product> productList = new ArrayList<>();

    public void populateProductList(final WebElement pProductSectionElement)
    {
        final List<WebElement> productElements = pProductSectionElement.findElements(By.xpath(ALL_PRODUCTS_PATTERN));

        for ( WebElement productElement : productElements ) {
            // test 1 - works 
            // System.out.println( "Product block: " + productElement.getText() );
            final Product product = new Product(productElement);
            // test 2 - wrong
            // System.out.println( "Title: " + product.getUrl() );
            // System.out.println( "Url: " + product.getTitle() );
            productList.add(product);
        }

        // test 3 - works
        //System.out.println(productElements.get(0).findElement(Product.URL_PATTERN).getAttribute("href"));
        //System.out.println(productElements.get(1).findElement(Product.URL_PATTERN).getAttribute("href"));
    }
}

class Product
{
    public final static String URL_PATTERN = "//div[@class='url']/a";
    public final static String TITLE_PATTERN = "//div[@class='title']";

    private String url;
    private String title;

    public Product(final WebElement productElement)
    {
        url = productElement.findElement(By.xpath(URL_PATTERN)).getAttribute("href");
        title = productElement.findElement(By.xpath(TITLE_PATTERN)).getText();
    }

    /* ... */
}

我正在努力解析的网页&#39;与Selenium有很多代码。我需要处理其中包含产品网格的一小部分。 致populateProductList()调用,我传递包含所有产品的DOM的结果部分。 (在Chrome中运行该XPath会返回预期的all_products节点。)

在该方法中,我将产品分成25个单独的Web元素,即产品块。 (这里我也确认它适用于Chrome并返回节点列表,每个节点都包含产品数据)

接下来,我想迭代结果列表,并将每个WebElement传递给Product()构造函数,为我初始化Product。 在我这样做之前,我进行了一个小测试并打印出产品块(参见测试1);在每个迭代中打印出单独的块。

执行产品分配后(再次在Chrome中确认xpath)我再次进行测试(参见测试2 )。

问题:这次测试只返回FIRST产品的网址/标题对进行EACH迭代。

除此之外,我尝试将Product的{​​{1}}次呼叫转移到循环中,但仍然存在同样的问题。接下来,我尝试运行findElement()并对结果执行findElement**s**();这次它正确返回了单个产品URL(参见测试3)。

然后当我在循环内的单个get(i).getAttribute("href")上执行findElements(URL_PATTERN)时,它会神奇地返回所有产品网址...这意味着productElement始终会返回第一个产品一套25个产品,而我希望findElement()只包含一个产品。

我认为这看起来像引用的问题,但我无法想出任何东西或在线找到解决方案。

对此有何帮助?谢谢!

java 1.7.0_15,Selenium 2.45.0和FF 37

1 个答案:

答案 0 :(得分:4)

问题出在产品定位器的XPATH中。

在selenium中的xpath表达式下面意味着您正在寻找文档中可以在任何地方的匹配元素。正如你想的那样,与父母不相关!!

//div[@class='url']/a

这就是为什么它总是返回相同的第一个元素。

因此,为了使它相对于父元素,它应该如下所示。 (只是一个。之前//)

public final static String URL_PATTERN = ".//div[@class='url']/a";
public final static String TITLE_PATTERN = ".//div[@class='title']";

现在,您可以搜索相对于父级的匹配子元素。

硒中的XPATH如下所示。

/a/b/c   --> Absolute - from the root

//a/b    --> Matching element which can be anywhere in the document (even outside the parent).

.//a/b   --> Matching element inside the given parent