我试图覆盖在子类中返回类型为list的方法,如下例所示,由于泛型问题,我无法在子类中执行此操作。我无法改变我的超类代码,所以我如何解决这个问题?任何人都可以指导我......非常感谢提前。
无法更新的超级课程
public class Animal {
private String legs;
}
public class TestSuper {
public List<Animal> addAnimals() {
return animals;
}
}
子类:
public class Dog extends Animal {
private String sound;
}
public class TestSub extends TestSuper {
public List<Dog> addAnimals() { --> I need to override this method but want return type as List of dogs
return null;
}
}
答案 0 :(得分:9)
如果超级班真的无法更新,我恐怕你根本就不能。
如果您能够更新超类:
在TestSuper
中,您可以使用public List<? extends Animal> addAnimals()
代替public List<Animal> addAnimals()
。
您有哪些其他解决方案:
您可以使用这样的实用方法:
@SuppressWarnings("unchecked")
public static <T extends Animal> List<T> cast(List<Animal> animals, Class<T> subclass) {
List<T> out = new ArrayList<T>();
for (Animal animal : animals) {
if (!subclass.isAssignableFrom(animal.getClass())) {
// the "animal" entry isn't an instance of "subclass"
// manage this case however you want ;)
} else {
out.add((T) animal);
}
}
return out;
}
然后,致电:
List<Dog> dogs = TheClassNameYouChose.cast(animals, Dog.class);
答案 1 :(得分:2)
您无法将List的泛型类型更改为该泛型类型的子类型。问题是对任何实例的引用都可以是该实例的子类型。那么让我们来看看这个例子
SuperType sup = new SubType();
sup.getAnimals().add(new Cat());
//adding cat is possible because getAnimals() returns List<Animal>
因此,您要将Cat
添加到仅包含Dogs
的列表中,这是错误的。
同样在重写方法中,您无法将通用类型的列表更改为具有该泛型类型的超类型的列表(如List<Dog>
到List<Animal>
)。为什么?让我们来看看这个例子
SuperType sup = new SubType();
Dog d = sup.getAnimals().get(0);
在子类型中,我们有动物列表,因此get(0)
可能会返回Cat。
如果您可以更改课程中的代码,请尝试使用此方法
class Super<T extends Animal> {
public List<T> addAnimals() {
return null;
}
}
class Sub extends Super<Dog> {
@Override
public List<Dog> addAnimals() {
return null;
}
}
答案 2 :(得分:1)
如果Dog
是Animal
的子类,则可以执行
以下签名:
public List<? extends Animal> addAnimals()
然后做
public List<Dog> addAnimals()
在子类中。
或者您可以创建超类通用<T>
和实现类<Dog>
答案 3 :(得分:0)
您需要使TestSuper使用泛型来真正使其正常工作。
TestSuper<T extends Animal> {
public List<T> addAnimals() {
}
}
TestSub extends TestSuper<Dog> {
}
答案 4 :(得分:0)
这是不可能的。超级合同说如果你有:
class Animal {}
class Dog extends Animal {}
class Cat extends Animal {}
以下代码必须有效:
TestSuper test = new AnySubclassOfTestSuper(); // including TestSub!
test.addAnimals().add(new Cat());
TestSub
类中的覆盖会违反此规则,因为您无法将Cat
添加到List<Dog>
。由于它违反了面向对象类型系统的一个相当基本的原则,你不会找到这种限制的完美解决方法。
将超类更改为使用List<? extends Animal>
的建议可行,因为您无法向该列表添加任何内容。