尝试在页面对象中使用静态方法时,我遇到了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类的实例时,您正在调用其构造函数:
NullPointerException
PageFactory
ComplaintPage
public ComplaintPage() { driver = Browser.extendedDriver(); PageFactory.initElements(driver, this); }
构造函数调用类的initElements方法PageFactory。此方法使用Java Reflection API 初始化所有WebElementand List<WebElement>字段。基本上,它将默认null值更改为接口的实现。它还提供了WebElement的惰性实例化,这意味着- WebElement仅在需要时-调用它们上的操作时才查找(寻找?)。
initElements
WebElement
List<WebElement
null
创建静态方法和静态WebElements时-您未调用类的构造函数,这导致未调用该initElements方法。
带有注释的所有元素@FindBy均未初始化。这就是为什么将PageFactory与静态方法一起使用不是一个好主意的原因。
@FindBy
除了使用PageFactory,您还可以driver.findElement在static方法中找到具有classic的元素。
driver.findElement
public void static setComplaintDate(WebDriver driver) { driver.findElement(By.cssSelector("[data-selector=date-received-complaint]")).sendKeys(LocalDate.now().toString()); }