class A {
public A get() { }
}
class B extends A {
}
返回类型与继承的方法不兼容, 如何解决这个问题?
答案 0 :(得分:1)
从JDK 5开始,Java允许您更改被重写方法的返回类型,只要新类型是原始类型的子类。 这称为covariant return type。 以下代码将正确编译:
class A {
A get() {
return new A();
}
void sayHello(){
System.out.println("Hello");
}
}
class B extends A {
@Override
B get() {
return new B();
}
void sayGoodbye(){
System.out.println("Goodbye");
}
}
class Test{
void check(){
B two=new B();
two.get().sayGoodbye();
}
}
请记住,重写方法的返回类型应该是返回类型的子类,以允许您在A变量上调用方法并获取有效的A对象(实际上是B实例):
void check(){
A two=new B();
A copy=two.get();
copy.sayHello();
}
答案 1 :(得分:0)
使用泛型:
class A<T extends A> {
public T get() { }
}
class B extends A<B> {
}
B b = new B();
B got = b.get();
答案 2 :(得分:0)
我假设你想写一些像B b = new B().get();
这样的东西,而不像Nikita Beloglazov那样明确的类型转换,尽管这不是Java支持的成语。 Eugene的答案运作得很好,尽管这仍然涉及到T
的强制转换并且还会产生一个丑陋的编译器警告。就个人而言,我会编写更像以下代码:
class A {
private A() {}
public A get() {
return new A();
}
}
class B extends A {
private B() {}
@Override
public A get() {
return new B();
}
public B getB() {
return (B) get();
}
}
你仍然无法做B b = new B().get();
,但是你可以做B b = new B().getB();
,这也很容易,而且更自我记录,因为你已经知道你想要一个{{1}对象,而不仅仅是任何B
对象。 A
会创建一个B对象,尽管只能访问A a = new B().get();
中声明的方法。