超类不会调用重写方法吗?

时间:2009-04-10 02:53:13

标签: java oop polymorphism override

我有以下课程:

class foo {
    public void a() {
        print("a");
    }
    public void b() {
        a();
    }
}

class bar extends foo {
    public void a() {
        print("overwritten a");
    }
}

当我现在调用bar.b()时,我希望它在foo中调用重写的方法a()。但是,它会打印“a”。

7 个答案:

答案 0 :(得分:10)

你的两个课程是否在不同的课程中?你的foo类方法是公共的,受保护的,私有的还是包本地的?显然,如果它们是私有的,这将不起作用。也许不那么明显,如果它们是本地包(即没有公共/受保护/私有范围),那么只有当你与原始类在同一个包中时才能覆盖它们。

例如:

package original;
public class Foo {
  void a() { System.out.println("A"); }
  public void b() { a(); }
}

package another;
public class Bar extends original.Foo {
  void a() { System.out.println("Overwritten A"); }
}

package another;
public class Program {
  public static void main(String[] args) {
    Bar bar = new Bar();
    bar.b();
  }
}

在这种情况下,你仍会得到'A'。如果您在Foo public或protected中声明原始a()方法,您将获得预期的结果。

答案 1 :(得分:9)

您可能正在尝试使用静态方法,这些方法无法正常工作,因为它们不会被覆盖。

一种好的检查方法是将@Override注释添加到bar.a()并查看编译器是否给出了一个错误,即a()实际上没有覆盖任何内容

答案 2 :(得分:7)

当我运行以下内容时:

public class Program {
    public static void main(String[] args) {
        bar b = new bar();
        b.b();
    }
}

class foo {
    public void a() {
       System.out.printf("a");
    }
    public void b() {
        a();
    }
}

class bar extends foo {
    public void a() {
        System.out.printf("overwritten a");
    }
}

我得到以下输出:

overwritten a

这是我期望看到的。

答案 3 :(得分:1)

这些方法是否定义为静态?这是我能看到得到这个结果的唯一方法。我在这里找到了一个很好的解释:http://faq.javaranch.com/view?OverridingVsHiding

答案 4 :(得分:0)

如果您来自C#或其他必须明确声明虚函数和/或覆盖函数的语言,您可能会感到困惑。

在Java中,所有实例函数都是虚拟的,并且可以被覆盖 - 除非它们被声明为私有和/或最终。

没有必要指定新的@Override注释来执行此操作,添加注释只是指定您的意图将被覆盖,并且如果它不是覆盖,将导致警告或错误。 (例如,如果您不小心拼错了方法名称。)

安德鲁的例子说明了这应该如何运作。

答案 5 :(得分:0)

从马的嘴里说:

http://download.oracle.com/javase/tutorial/java/IandI/override.html

“被调用的被重写方法的版本是子类中的版本。被调用的隐藏方法的版本取决于它是从超类还是从子类调用。”

因此,如果它们都是静态方法并且您从超类调用该方法,则调用超类方法,而不是子类方法。所以真的,没有压倒一切。

答案 6 :(得分:-1)

当我在开发Android程序时,我的Java类遇到了同样的问题。事实证明,问题不在于类,而在于Android框架处理屏幕输出的方式。

例如,如果输出是在父类的onCreate()方法中编程的,则Android无法从第一个子节点之外的子类中正确获取重写方法的输出。老实说,我不了解整个方法调用顺序。

要解决这个问题,我只需在onResume()中编写输出,现在它似乎工作正常。