当前位置: 首页 > 面试题库 >

Java 8接口方法中不允许“同步”的原因是什么?

宇文曦
2023-03-14
问题内容

在Java 8中,我可以轻松地编写:

interface Interface1 {
    default void method1() {
        synchronized (this) {
            // Something
        }
    }

    static void method2() {
        synchronized (Interface1.class) {
            // Something
        }
    }
}

我将获得在类中也可以使用的完整同步语义。但是,我不能synchronized在方法声明上使用修饰符:

interface Interface2 {
    default synchronized void method1() {
        //  ^^^^^^^^^^^^ Modifier 'synchronized' not allowed here
    }

    static synchronized void method2() {
        // ^^^^^^^^^^^^ Modifier 'synchronized' not allowed here
    }
}

现在,可以争论说这两个接口的行为方式相同,只是在on 和on上Interface2建立了一个 契约 ,这比 契约
要强一点。当然,我们也可能会争辩说,实现不应对具体的实现状态做出任何假设,或者这样的关键字根本无法发挥作用。method1()``method2()``Interface1``default

题:

JSR-335专家组决定不支持synchronized接口方法的原因是什么?


问题答案:

虽然乍一看似乎很想synchronized在默认方法上支持修饰符,但事实证明这样做很危险,因此被禁止。

同步方法是一种方法的简写,该方法的行为就像将整个主体包围在一个synchronized块中,该块的锁定对象是接收者。将这种语义扩展到默认方法似乎也很明智。毕竟,它们也是带有接收器的实例方法。(请注意,synchronized方法完全是语法优化;不需要它们,它们比相应的synchronized块更紧凑。有一个合理的论点是,这首先是一个过早的语法优化,并且同步方法造成的问题比他们解决的问题还多,但那艘船航行了很久。)

那么,为什么会有危险呢?同步与锁定有关。锁定是关于协调对可变状态的共享访问。每个对象都应具有一个同步策略,该策略可以确定哪些锁可以保护哪些状态变量。(请参阅《Java并发实践》第2.4节。)

许多对象将 Java监视器模式 (JCiP
4.1)用作其同步策略,其中对象的状态由其内部锁定来保护。这种模式没有什么魔术或特殊之处,但是它很方便,并且synchronized在方法上使用关键字隐式地采用了这种模式。

拥有状态的类就是确定该对象的同步策略的类。但是接口并不拥有它们所混入的对象的状态。因此,在接口中使用同步方法假设一种特定的同步策略,但是您没有合理的假设基础,因此很可能是这样使用同步不会提供任何额外的线程安全性(您可能在错误的锁上进行同步)。这会给您一种错误的信心,即您对线程安全已做过一些事情,并且没有错误消息告诉您您采用了错误的同步策略。

很难始终如一地维护单个源文件的同步策略。确保子类正确遵守其父类定义的同步策略甚至更加困难。在这样的松散耦合的类(一个接口以及可能实现它的许多类)之间尝试这样做几乎是不可能的,而且容易出错。

考虑到所有这些反对意见,那将是什么理由呢?似乎它们主要是关于使接口的行为更像特征。尽管这是可以理解的愿望,但是默认方法的设计中心是接口演变,而不是“特质”。我们力求做到这两个方面能够始终如一地实现,但是当一个与另一个发生冲突时,我们必须选择支持主要的设计目标。



 类似资料:
  • 虽然有些人曾经问过这个问题,但那是在Java8发布之前。 以前,静态成员是不允许的,因为实现细节不能在接口中定义。这也是为什么任何东西都不应该是私有的,因为接口的实现者需要提供实现细节。 Java8改变了这种情况,不是吗?默认方法定义实现细节,静态方法也是如此。那么,为什么仍然不允许这样做呢?

  • 问题内容: 但是那里给出的解决方案不起作用。他们说我需要采取以下行动: 在项目结构中| 在“项目”对话框中,在界面中将“项目语言级别”更改为6.0-@Override。 但是,目前项目语言级别是6.0,但是我仍然看到错误。 维克(Vic),这是一个窗口,并且在语言级(Language level)下没有JVM版本(不幸的是,由于我有10个信誉,所以我无法发布图像) 问题答案: 如果您的项目有多个模

  • 问题内容: Java 8最有用的功能之一是接口上的新方法。引入它们的原因基本上有两个(可能还有其他原因): 提供实际的默认实现。例: 允许JDK API演进。例: 从API设计人员的角度来看,我希望能够在接口方法上使用其他修饰符,例如。在添加便捷方法时,这将很有用,以防止在实现类时“意外”覆盖: 如果已经上过课,以上是已经很普遍的做法: 现在,并且显然是相互矛盾的关键字,但是默认关键字本身并没有严

  • 我们举个例子: 一个常见的解决方案是转移到一个抽象类,但是在我的具体案例中,我有一个枚举的接口,所以在这里不适用。我想这不是被忽略了,就是因为接口背后的原始想法,即它们是可用方法的“契约”,但我想我需要关于这是怎么回事的输入。 我读过“为什么Java 8接口方法中不允许使用”final“?”,其中说: 默认方法的基本思想是:它是具有默认实现的接口方法,派生类可以提供更具体的实现 与关联问题一样,由

  • 问题内容: 为什么在Java 8中允许接口具有方法? 如下面的代码所述,它可以正常工作并正确产生输出。 目前它的行为就像一个类,我已经用main方法执行了。 我们为什么需要这个? 问题答案: 从Java 8开始,接口中允许使用静态方法。 main()是静态方法。 因此,在接口中允许main()。 我们 不需要 它,因为以前不允许这样做,但是我们幸存了下来。但是,根据定义,由于静态方法不是绑定到类的