【问题标题】:Why `private static` field is not allowed in Java 8 interface?为什么 Java 8 接口中不允许使用“私有静态”字段?
【发布时间】:2015-07-16 10:50:48
【问题描述】:

当我试图编译以下代码时

public interface SomeInterface{
    private static Logger logger = Logger.getLogger();

    public default void someMethod(){
        logger.info("someMethod: default implementation");
    }
}

我收到一个错误

Illegal modifier for the interface field SomeInterface.logger; only public, static & final are permitted

当我删除 private 修饰符时,代码会编译,但我不希望包中的其他类看到此字段。

为什么当 Java 确实有意义时却不允许我做这样的事情?

【问题讨论】:

  • 如果重写 someMethod 方法会怎样?
  • @Stultuske 我将提供新的实现。如果我不覆盖它会怎样?
  • 从逻辑上讲,接口旨在提供一种概括几个不同类的行为的方法。那就是 - 用户必须做什么来控制该类的行为。以汽车(在现实生活中)为例:您有车轮、收音机、制动器、离合器等。您了解这些并学习如何操作它们。但是如果接口包含一些你无法访问的东西——你将如何使用它来控制类的行为?换句话说 - 接口是你通往世界的门户 - 使用它们来概括而不是隐藏字段等......
  • @Mackiavelli 但默认实现需要此私有字段。接口的本意是不提供任何实现,但 Java 8 稍微改变了这一点。
  • 我知道人们会说这是不正确的,但在某种程度上,接口越来越接近 cpp 抽象类。为什么?我不知道。但是实施应该注意这一点。实现接口的用户根本不应该查看私有内容。

标签: java interface java-8 private-members


【解决方案1】:

在 Java-8 之前的世界观中,接口纯粹是为了接口契约,而私有成员的存在纯粹是为了实现,所以这种限制是完全合理的。

在后 Java-8 世界中,接口可以承载行为(但不能承载状态),因此开始合理地询问类的其他特性是否也应该应用于接口。 (然而,仅仅因为某些东西可能是“合理的”并不意味着它必须得到支持;通常有不止一种合理的方式来构建世界。)

在 Java 9 中,将支持接口中的私有方法。

【讨论】:

  • 这个答案最好地解释了我的问题 :) 你有参考 Java 9 中的 private 成员吗?
  • kamil09875:openjdk.java.net/jeps/213(第 5 点);如果你认为,这回答了你的问题,你可以accept回答。
  • “在 Java 9 中,将支持接口中的私有成员” - 似乎只支持 私有方法,但不支持 私有静态最终字段我>?你有任何关于为什么私有静态最终字段被遗漏的信息吗?
  • 他们从来没有进去过。我应该在上面说“方法”,因为这一直是计划;认为这是一个错字。 (答案已被编辑。)
  • 为什么忽略了私有静态最终字段?
【解决方案2】:

接口不是类。他们没有私人国家。即使是界面中的公共记录器也是一种设计气味和对界面的滥用。

接口中静态字段的用例主要用于编译时常量,而不是有状态对象。

【讨论】:

  • 是的,我不认为这是“设计气味”。
【解决方案3】:

接口的目标是定义由其他类实现的东西。私有字段没有定义任何东西,因为它在界面之外是不可见的。因此,它在此构造中没有任何意义。如何使用它可能有一些技巧(可能来自接口内部类),但无论如何看起来都不是一个好的设计。

如果您实际实现了部分功能,请改用抽象类。

【讨论】:

  • 默认实现在界面外也不可见
  • 在同一个包中可见。
  • 如何在同一个包中可见? Java 8 接口也不同,因为它们确实实现了部分功能。如果它们不是私有静态字段,则根本不需要
  • 那么为什么我们在 Java 9 中有私有方法呢?考虑您的“私有字段没有定义任何内容,因为它在界面外不可见。因此它在此构造中没有任何意义。”
  • @LiSeeLeiCow-Q__Q 私有方法有助于实现默认方法(例如,防止代码重复)。与私有静态方法 ↔ 公共静态方法相同。
【解决方案4】:

接口就像任何类的蓝图,您可以在其中声明您的成员。任何实现该接口的类都对其定义负责。 私有成员只能被同一个类成员访问,这在接口方面没有意义。 受保护的成员可以被同一个类成员和继承的类成员访问,但是在接口的情况下,我们从不扩展接口,我们实现它。所以任何接口一般只能包含公共方法,

【讨论】:

    【解决方案5】:
    
    public interface SomeInterface {
        public default void someMethod() {
            SomeInterfaceInternal.logger.info("someMethod: default implementation");
        }
    }
    
    final class SomeInterfaceInternal {
        protected static final Logger logger = LoggerFactory.getLogger(SomeInterface.class);
    }
    
    

    【讨论】:

    • 这和原来的包私有字段有同样的问题,即包中的类可以访问它
    • @Enerccio 实际上你从来没有一个“包私有字段”。如果您删除接口成员上的修饰符,您将获得一个公共成员,而不是包私有成员。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2014-06-20
    • 1970-01-01
    • 2016-10-06
    • 2013-11-12
    相关资源
    最近更新 更多