0

我对 Java 类实现的抽象方法符号感到很困惑。

例如,考虑:

interface Programmer {
    Object program();
}

class Linus implements Programmer {
    public String program() {
        return "goto end;";
    }
}

public class Main {
    public static void main(String[] args) {
        System.out.println(new Linus().program());
    }
}

这显然是可以接受的,因为任何期望来自 Linus.program() 的对象都会得到一个(特别是字符串)。

但现在考虑一下:

interface OS {
    void run(String code);
}

class Linux implements OS {
    public void run(Object code) {
        System.out.println("Hello world");
    }
}

public class Main {
    public static void main(String[] args) {
    new Linux().run("print 'Hello world'");
    }
}

这无法编译,并产生错误:

The type Linux must implement the inherited abstract method OS.run(String)

现在,任何东西都可以将字符串传递到操作系统接口的任何实例中,你当然可以在 Linux 上做到这一点。

我看不出第二个无法编译的原因。在让 Java 编译第二个程序的后果中,我是否遗漏了一些使这令人望而却步的东西?

4

3 回答 3

5

您为第二种情况所做的是方法重载。
方法的签名是“方法名称”和“参数的数量和类型”
在第二种情况下,您正在更改编译器的参数类型是另一种方法,并要求您实现抽象(未实现)的方法。如果您实现了接口中存在的方法,public void run(String code)那么该方法public void run(Object code)将被视为重载方法。

注意:方法覆盖从不基于方法的返回类型。由于返回类型不被视为方法的签名

于 2013-03-09T09:06:24.453 回答
0

您不能传递基类 Object 的实例来代替 String 类的实例,但反之亦然。在您的界面操作系统中,run 方法的参数是 String 类型,而在实现类中,您将参数概括为 Object 类型,这是不允许的。因此,编译器抱怨。

于 2013-03-09T09:09:44.890 回答
0

你说“现在,任何东西都可以将字符串传递给操作系统接口的任何实例,你当然可以在 Linux 上做到这一点。” => 问题是你可以,但不是必须的。

考虑一下:由于该方法的 Linux 版本将 Object 作为参数,因此您可以尝试传递一个 Collection,例如。因此,您不会尊重接口,它明确地将参数限制为字符串。这就是为什么 Linux 版本的方法不被认为是 OS 版本的“子”。

由于接口中的每个方法都需要有它的“子”,编译器会抱怨你没有为 OS.run(String) 实现一个“子”

于 2013-03-09T09:18:01.120 回答