package package1;
public class TestPackage1 {
String getString(){
return "This is package1 class." ;
}
}
package package2;
import package1.TestPackage1;
public class TestPackage2 extends TestPackage1 {
public String getString(){
return "This is package2 class.";
}
}
package package1;
import package2.TestPackage2;
public class Test {
public static void main(String[] args) {
TestPackage1 test = new TestPackage2();
System.out.println(test.getString());
}
}
十分简单,第一眼看上去,你会觉得 TestPackage2 继承了 TestPackage1 并且重写 getString()方法。
请注意,测试程序和TestPackage2 在同一个包里,从理论上说,这个测试程序应该输出“This is package2 class.”, 因为test的实体是一个TestPackage2对象而不是TestPackage1。 所以,当我们调用test.getString()时,真正被调用的应该是TestPackage2里的getString()。可是事实如何呢?输出是 “This is package1 class.”,为什么会这样的?
原因很简单,因为在TestPackage1里方法getString()的标签(signature)是 “default”,是默认的不用写出来。这导致这个方法只能在这个包里面可见。TestPackage2虽然继承了TestPackage1,却没有办法“看见”getString()方法,因为TestPackage2在另外一个包里。所以当我们在测试程序里调用test.getString(),程序首先会寻找TestPackage2中是否重写了这个方法,在这里请一定要注意标签是default的,当然是没有发现。结果程序就会调用父类的相应方法,故父类中的结果就被输出了。
解决的办法很简单,但凡是要被重写的方法一定不能定义成“default”,最少要定义成 “protected”。
如果你是在使用Eclipse的话,你可以在eclipse中进行设置,把这种情况视为error就可以避免这种错误的产生。方法是:winodw à preferences à style à methods overridden but not package visible 选项设为:Error。