Java 和 Selenium:页面对象中的静态方法
Java and Selenium: Static methods in Page Objects
当我尝试在页面对象中使用静态方法时遇到 NullPointerExceptions 问题。如果我用非静态方法来做,它工作正常。
非静态版本:
public class ComplaintPage {
private ExtendedWebDriver driver;
@FindBy(css = "[data-selector=date-received-complaint]")
public WebElement dateComplaintReceoved;
public ComplaintPage() {
driver = Browser.extendedDriver();
PageFactory.initElements(driver, this);
}
public void setComplaintDate() {
dateComplaintReceoved.sendKeys(LocalDate.now().toString());
}
}
Calling code:
ComplaintPage complaintPage = new ComplaintPage;
complaintPage.setComplaintDate();
这很好用。日期字段已设置。
静态版
public class ComplaintPage {
private static ExtendedWebDriver driver;
@FindBy(css = "[data-selector=date-received-complaint]")
public static WebElement dateComplaintReceoved;
public ComplaintPage() {
driver = Browser.extendedDriver();
PageFactory.initElements(driver, this);
}
public void static setComplaintDate() {
* dateComplaintReceoved.sendKeys(LocalDate.now().toString());
}
}
Calling code:
ComplaintPage.setComplaintDate();
这不起作用,并导致在标有“*”的行(访问 WebElement 的行)上出现 java.lang.NullPointerException。
我有点喜欢在测试中使用这样的静态方法,因为我没有真正发现它有什么问题,而且它使代码更易于阅读。我以前在 C#/VS 中做过,但出于某种原因,我在这里遗漏了一些重要的东西。
NullPointerException
由于 PageFactory
的工作方式而被抛出。你看,当你创建 ComplaintPage
class 的实例时,你正在调用它的构造函数:
public ComplaintPage() {
driver = Browser.extendedDriver();
PageFactory.initElements(driver, this);
}
构造函数调用PageFactory
class的initElements
方法。此方法使用 Java 反射 API 初始化所有 WebElement
和 List<WebElement
> 字段。它基本上将默认 null
值更改为接口的实现。它还提供了 WebElement 的惰性实例化,这意味着 - WebElement
s 仅在需要时被发现(寻找?) - 当您调用对它们的操作时。
当您创建静态方法和静态 WebElements 时 - 您没有调用 class 的构造函数,这导致未调用 initElements
方法。
所有用@FindBy
注释的元素都没有初始化。这就是为什么将 PageFactory 与静态方法一起使用不是一个好主意。
除了使用 PageFactory,您还可以在静态方法中找到带有 classic driver.findElement
的元素。
public void static setComplaintDate(WebDriver driver) {
driver.findElement(By.cssSelector("[data-selector=date-received-complaint]")).sendKeys(LocalDate.now().toString());
}
当我尝试在页面对象中使用静态方法时遇到 NullPointerExceptions 问题。如果我用非静态方法来做,它工作正常。
非静态版本:
public class ComplaintPage {
private ExtendedWebDriver driver;
@FindBy(css = "[data-selector=date-received-complaint]")
public WebElement dateComplaintReceoved;
public ComplaintPage() {
driver = Browser.extendedDriver();
PageFactory.initElements(driver, this);
}
public void setComplaintDate() {
dateComplaintReceoved.sendKeys(LocalDate.now().toString());
}
}
Calling code:
ComplaintPage complaintPage = new ComplaintPage;
complaintPage.setComplaintDate();
这很好用。日期字段已设置。
静态版
public class ComplaintPage {
private static ExtendedWebDriver driver;
@FindBy(css = "[data-selector=date-received-complaint]")
public static WebElement dateComplaintReceoved;
public ComplaintPage() {
driver = Browser.extendedDriver();
PageFactory.initElements(driver, this);
}
public void static setComplaintDate() {
* dateComplaintReceoved.sendKeys(LocalDate.now().toString());
}
}
Calling code:
ComplaintPage.setComplaintDate();
这不起作用,并导致在标有“*”的行(访问 WebElement 的行)上出现 java.lang.NullPointerException。
我有点喜欢在测试中使用这样的静态方法,因为我没有真正发现它有什么问题,而且它使代码更易于阅读。我以前在 C#/VS 中做过,但出于某种原因,我在这里遗漏了一些重要的东西。
NullPointerException
由于 PageFactory
的工作方式而被抛出。你看,当你创建 ComplaintPage
class 的实例时,你正在调用它的构造函数:
public ComplaintPage() {
driver = Browser.extendedDriver();
PageFactory.initElements(driver, this);
}
构造函数调用PageFactory
class的initElements
方法。此方法使用 Java 反射 API 初始化所有 WebElement
和 List<WebElement
> 字段。它基本上将默认 null
值更改为接口的实现。它还提供了 WebElement 的惰性实例化,这意味着 - WebElement
s 仅在需要时被发现(寻找?) - 当您调用对它们的操作时。
当您创建静态方法和静态 WebElements 时 - 您没有调用 class 的构造函数,这导致未调用 initElements
方法。
所有用@FindBy
注释的元素都没有初始化。这就是为什么将 PageFactory 与静态方法一起使用不是一个好主意。
除了使用 PageFactory,您还可以在静态方法中找到带有 classic driver.findElement
的元素。
public void static setComplaintDate(WebDriver driver) {
driver.findElement(By.cssSelector("[data-selector=date-received-complaint]")).sendKeys(LocalDate.now().toString());
}