绑定不匹配错误和Java通用方法
问题内容:
我收到以下错误:
Bound mismatch: The generic method constructPage(WebDriver, int, Class<T>) of type
Page<T> is not applicable for the arguments (WebDriver, int, Class<HomePage>). The
inferred type HomePage is not a valid substitute for the bounded parameter <T extends
Page<T>>
我正在尝试登录并返回一个HomePage(如果成功),以及一个LoginPage(如果不使用泛型)。
我有一个基类,Page,由SecuredPage扩展,用于登录墙后面的页面。我写了一个通用的辅助方法,可以构造任何类型的页面。LoginPage上的登录方法使用此方法。LoginPage扩展Page,而HomePage扩展SecuredPage。SecuredPage扩展了Page。如果返回LoginPage,则登录方法有效,但是在尝试返回HomePage时出现上述错误。由于HomePage是页面的子类,因为它的父类扩展了Page,所以我对为什么HomePage不是边界参数的有效替代感到困惑<T extends Page<T>>
。
public abstract class Page<T extends Page<T>> extends SlowLoadableComponent<T> {
protected static final <T extends Page<T>> T constructPage(WebDriver driver,
int timeoutInSeconds, java.lang.Class<T> pageClass)
{
Page<T> p = null;
try {
Constructor<T> pageConstructor = pageClass.getConstructor(
WebDriver.class, String.class, Integer.TYPE);
p = pageConstructor.newInstance(driver, driver.getCurrentUrl(),
timeoutInSeconds);
p.get();
} catch(Exception e) {
}
return pageClass.cast(p);
}
}
这是SecuredPage类:
public class SecuredPage extends Page<SecuredPage> {
.....
}
这是主页:
public final class HomePage extends SecuredPage {
......
}
这是LoginPage:
public final class LoginPage extends Page<LoginPage> {
public final HomePage loginWithGoodCredentials(final User user) {
return login(user, HomePage.class);
}
public final LoginPage loginWithBadCredentials(final User user) {
return login(user, LoginPage.class);
}
public final <T extends Page<T>> T login(final User user, final Class<T>
expectedPage) {
enterUsername(user.getUsername());
enterPassword(user.getPassword());
loginButton.click();
return Page.constructPage(getDriver(), getTimeoutInSeconds(),
expectedPage);
}
}
问题答案:
问题是那HomePage
是一个Page<SecuredPage>
而不是一个Page<HomePage>
。该login
方法Page<HomePage>
将从其通用签名返回a
。
您必须将的一般参数HomePage
设为与自身相关,而不是SecuredPage
。这样可以解决编译器错误。保持SecuredPage
通用性,但确保其边界扩展SecuredPage<T>
。然后分配HomePage
本身泛型参数T
在HomePage
。
class SecuredPage<T extends SecuredPage<T>> extends Page<T> {
...
}
class HomePage extends SecuredPage<HomePage> {
...
}