栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 面试经验 > 面试问答

何时使用默认方法初始化接口?

面试问答 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

何时使用默认方法初始化接口?

这是一个非常有趣的问题!

似乎JLS第12.4.1节应该明确地涵盖这一点。但是,OracleJDK和OpenJDK(javac和HotSpot)的行为与此处指定的行为不同。特别是,本节的示例12.4.1-3涵盖了接口初始化。示例如下:

interface I {    int i = 1, ii = Test.out("ii", 2);}interface J extends I {    int j = Test.out("j", 3), jj = Test.out("jj", 4);}interface K extends J {    int k = Test.out("k", 5);}class Test {    public static void main(String[] args) {        System.out.println(J.i);        System.out.println(K.j);    }    static int out(String s, int i) {        System.out.println(s + "=" + i);        return i;    }}

其预期输出为:

1j=3jj=43

实际上我得到了预期的输出。但是,如果将默认方法添加到interface

I

interface I {    int i = 1, ii = Test.out("ii", 2);    default void method() { } // causes initialization!}

输出更改为:

1ii=2j=3jj=43

这清楚地表明接口

I
正在被初始化!仅使用默认方法就足以触发初始化。默认方法不必被调用,重写或提及,抽象方法的存在也不会触发初始化。

我的猜测是,HotSpot实现希望避免在

invokevirtual
调用的关键路径中添加类/接口初始化检查。在Java
8和默认方法之前,
invokevirtual
永远不会最终在接口中执行代码,因此这没有发生。可能有人认为这是类/接口准备阶段(JLS12.3.2)的一部分,该阶段初始化诸如方法表之类的东西。但这可能太过分了,而意外地进行了完全初始化。

我在OpenJDK编译器-
dev邮件列表中提出了这个问题。AlexBuckley(JLS的编辑)收到了答复,他提出了更多针对JVM和lambda实现团队的问题。他还指出,规范中存在一个错误,该错误表明“
T是一个类,并且调用了T声明的静态方法”,如果T是接口,则也应适用。因此,这里可能同时存在规范和HotSpot错误。

披露 :我在OpenJDK上为Oracle工作。如果人们认为这使我在悬赏这个问题上获得了不公平的优势,那么我愿意对此灵活。



转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/437584.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号