Method chaining + inheritance don’t play well together?
这个问题是在C++环境中提出的,但我对Java很好奇。对虚拟方法的关注并不适用(我认为),但是如果您遇到这种情况:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 | abstract class Pet { private String name; public Pet setName(String name) { this.name = name; return this; } } class Cat extends Pet { public Cat catchMice() { System.out.println("I caught a mouse!"); return this; } } class Dog extends Pet { public Dog catchFrisbee() { System.out.println("I caught a frisbee!"); return this; } } class Bird extends Pet { public Bird layEgg() { ... return this; } } { Cat c = new Cat(); c.setName("Morris").catchMice(); // error! setName returns Pet, not Cat Dog d = new Dog(); d.setName("Snoopy").catchFrisbee(); // error! setName returns Pet, not Dog Bird b = new Bird(); b.setName("Tweety").layEgg(); // error! setName returns Pet, not Bird } |
在这种类层次结构中,是否有任何方法可以返回
如果希望避免编译器发出未选中的强制转换警告(并且不希望@suppresswarnings("unchecked"),则需要执行以下操作:
首先,您对宠物的定义必须是自引用的,因为宠物总是一个通用类型:
1 | abstract class Pet <T extends Pet<T>> |
其次,setname中的
The"getThis" trick provides a way to
recover the exact type of the this
reference.
这将导致下面的代码编译并运行,而不发出警告。如果您想扩展子类,那么该技术仍然有效(尽管您可能需要将中间类泛型化)。
产生的代码是:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 | public class TestClass { static abstract class Pet <T extends Pet<T>> { private String name; protected abstract T getThis(); public T setName(String name) { this.name = name; return getThis(); } } static class Cat extends Pet<Cat> { @Override protected Cat getThis() { return this; } public Cat catchMice() { System.out.println("I caught a mouse!"); return getThis(); } } static class Dog extends Pet<Dog> { @Override protected Dog getThis() { return this; } public Dog catchFrisbee() { System.out.println("I caught a frisbee!"); return getThis(); } } public static void main(String[] args) { Cat c = new Cat(); c.setName("Morris").catchMice(); Dog d = new Dog(); d.setName("Snoopy").catchFrisbee(); } } |
这个老把戏怎么样:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
不,不是真的。您可以通过使用协变返回类型来解决这个问题(感谢McDowell提供正确的名称):
1 2 3 4 5 |
(协变返回类型仅在Java 5和以上,如果这是您的关注点)。
这有点复杂,但是你可以用泛型来实现:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 | abstract class Pet< T extends Pet > { private String name; public T setName( String name ) { this.name = name; return (T)this; } public static class Cat extends Pet< Cat > { public Cat catchMice() { System.out.println("I caught a mouse!" ); return this; } } public static class Dog extends Pet< Dog > { public Dog catchFrisbee() { System.out.println("I caught a frisbee!" ); return this; } } public static void main (String[] args){ Cat c = new Cat(); c.setName("Morris" ).catchMice(); // error! setName returns Pet, not Cat Dog d = new Dog(); d.setName("Snoopy" ).catchFrisbee(); // error! setName returns Pet, not Dog } } |
1 2 3 4 5 6 7 |
和
1 2 3 4 5 6 7 | public class Cat extends Pet<Cat> { public Cat catchMice() {return this;} public static void main(String[] args) { Cat c = new Cat().setName("bob").catchMice(); } |
}