83

我知道接口就像一个 100% 纯抽象类。所以,它不能有方法实现。但是,我看到了一个奇怪的代码。谁能解释一下?

代码片段:

 interface Whoa {
        public static void doStuff() {
            System.out.println("This is not default implementation");
        }
 }

编辑:

我的 IDE 是 Intellij Idea 13.1。项目 SDK 为 java 7 <1.7.0_25>。IDE 未显示任何编译器错误。但是,当我在命令行编译代码时,我收到以下消息。

Whoa.java:2: error: modifier static not allowed here
    public static void doStuff() {
                       ^
4

3 回答 3

111

Java 8开始,除了默认方法之外,您还可以在接口中定义静态方法。

  • 静态方法是与定义它的类相关联的方法,而不是与任何对象相关联的方法。类的每个实例都共享其静态方法。

  • 这使您可以更轻松地在库中组织辅助方法;您可以将特定于接口的静态方法保留在同一接口中而不是单独的类中。

  • 以下示例定义了一个静态方法,该方法检索ZoneId与时区标识符对应的对象;如果没有ZoneId与给定标识符对应的对象,它将使用系统默认时区。(因此,您可以简化方法getZonedDateTime

这是代码:

public interface TimeClient {
   // ...
    static public ZoneId getZoneId (String zoneString) {
        try {
            return ZoneId.of(zoneString);
        } catch (DateTimeException e) {
            System.err.println("Invalid time zone: " + zoneString +"; using default time zone instead.");
            return ZoneId.systemDefault();
        }
    }

   default public ZonedDateTime getZonedDateTime(String zoneString) {
      return ZonedDateTime.of(getLocalDateTime(), getZoneId(zoneString));
   }    
}

也可以看看

于 2014-03-28T13:05:15.593 回答
20

这仅在 Java 8 中才有可能。在Java 7语言规范 §9.4中,它明确指出:

如果在接口中声明的方法被声明为静态,这是编译时错误,因为静态方法不能是抽象的。

所以在 Java 7 中,接口中的静态方法不能存在。

如果您转到Java 8语言规范 §9.4.3,您会看到它说:

静态方法也有一个块体,它提供了方法的实现。

所以它明确指出在 Java 8 中,它们可以存在。

我什至尝试在 Java 1.7.0_45 中运行您的确切代码,但它给了我错误“此处不允许使用修饰符静态”。


这是直接来自 Java 8教程Default Methods (Learning the Java Language > Interfaces and Inheritance)的引用:

静态方法

除了默认方法外,您还可以在接口中定义静态方法。(静态方法是与定义它的类相关联的方法,而不是与任何对象相关联。类的每个实例都共享其静态方法。)这使您可以更轻松地在库中组织辅助方法;您可以将特定于接口的静态方法保留在同一接口中而不是单独的类中。以下示例定义了一个静态方法,该方法检索ZoneId与时区标识符对应的对象;如果没有 ZoneId与给定标识符对应的对象,它将使用系统默认时区。(因此,您可以简化方法getZonedDateTime):

public interface TimeClient {
    // ...
    static public ZoneId getZoneId (String zoneString) {
        try {
            return ZoneId.of(zoneString);
        } catch (DateTimeException e) {
            System.err.println("Invalid time zone: " + zoneString +
                "; using default time zone instead.");
            return ZoneId.systemDefault();
        }
    }

    default public ZonedDateTime getZonedDateTime(String zoneString) {
        return ZonedDateTime.of(getLocalDateTime(), getZoneId(zoneString));
    }    
}

与类中的静态方法一样,您可以将接口中的方法定义指定为静态方法,并static在方法签名的开头使用关键字。接口中的所有方法声明,包括静态方法,都是隐式的public,所以你可以省略public修饰符。

于 2014-03-28T13:24:28.633 回答
15

对于java版本7 或更低版本,您可以使用在接口主体中声明的嵌套类来实现类似的功能。这个嵌套类实现了外部接口。

例子:

interface I1{
    public void doSmth();

    class DefaultRealizationClass implements  I1{

        @Override
        public void doSmth() {
           System.out.println("default realization");
        }
    }
}

我们如何在代码中使用它?

class MyClass implements I1{

    @Override
    public void doSmth() {
         new I1.DefaultRealizationClass().doSmth();
    }   
}

因此,默认实现封装在接口中。

于 2014-03-28T19:05:54.203 回答