在进行面试时,我发现大多数应聘者都不知道Java中的private修饰符真正意味着什么。 他们知道一些足以进行日常编码的事情,但还远远不够。 这不成问题。 足够了解就足够了。 但是,了解Java的一些内部工作仍然很有趣。 在极少数情况下,它可能会揭示某些细节。 如果没有别的,那就很有趣.orElse(whyDoYouReadIt)
?
顺便说一句:在我看来,即使与我个人有关的陈述和含义是虚假的,提及面谈也是写评委的好机会。 毕竟,我这个人并不重要,并且远离批评我的事实,我发现这篇文章很有趣,关于访谈的结论很重要,实际上完全符合我的观点。
本文希望以比阅读语言标准更具可读性的方式描述一些Java事实。
那么什么是私人的?
private
是Java中的访问修饰符。 如果您具有某个类的private
成员(方法,字段,内部或嵌套类或嵌套接口),则该类只能由同一类中的代码使用。 有趣的问题是:当private
方法包含多个类时,会发生什么? 怎么可能不止一个班? 如果有一个包含另一个类的类,并且在内部/嵌套类内部有一个private
方法,则它在内部/嵌套类内部以及顶级类中。
封闭类内部的私有方法可以从外部类调用吗? 封闭类中的代码可以在外部类中调用私有方法吗? 在两种情况下答案都是肯定的。 样例代码
package javax0.package1;class TopLevelClass {void topMethod(){NestedClass nc = new NestedClass();nc.method();}private int z;interface NestedInterface {default void method(){TopLevelClass tlc = new TopLevelClass();tlc.z++;}}static class NestedClass {private int k;private void method() {TopLevelClass tlc = new TopLevelClass();k = tlc.z;}}
}
清楚地显示了这种情况:嵌套类NestedClass
和嵌套接口NestedInterface
都包含可以访问包含private
字段z
的外部类的代码。 同样,顶级类代码可以在嵌套类内部调用私有方法。 在这种情况下,此示例代码实际上并未执行任何合理的操作并不重要。
如果我们编译此单个源文件,则会得到三个类文件:
-
TopLevelClass$NestedClass.class
-
TopLevelClass$NestedInterface.class
-
TopLevelClass.class
那是因为JVM不知道什么是顶层和嵌套的。 JVM对嵌套类和顶级类一无所知。 对于JVM,类只是一个类。 如果您坚持的话,要参加顶级课程。 这主要是因为Java语言1.0没有嵌套类和内部类,并且JVM是根据该类设计的。 在Java 1.1中引入内部类和嵌套类时,仅修改了编译而不是JVM,因此内部类和嵌套类仍然是语言功能,但不能直接由JVM处理。
顶级类如何访问嵌套在源代码中的另一个类中的私有方法,但是在编译时,它只是另一个“顶级”类。 它们处于同一级别。 如果将可访问性更改为公共,那么我们也可以从其他类中访问它,但我们不能。 编译器将不允许其他类中的任何其他代码访问私有方法,即使我们做了一些技巧来克服编译器,生成的类填充也将使JVM引发异常。 Java中的私有是私有的。
真正发生的是,编译器生成了特殊的getter和setter方法来访问字段z
。
将为从同一顶级类内部的不同类访问的每个私有字段或方法创建一个桥接方法。 如果未从封闭类访问私有对象,则不会生成该方法。 如果仅读取该字段,则仅生成getter;如果仅从外部进行设置,则仅生成setter。
这也是一个有趣的失败,因为认为只能从同一对象内访问私有字段(或其他任何内容)。 这是我们在编程时使用这些成员的通常方式,但是如果代码具有对相同类型另一个实例的引用,则通过该引用,我们可以访问其他对象的私有字段,就像访问“我们的对象”一样好。自己的”字段。 这是罕见的情况吗? 您可能会想,因为您很少对其进行编程。 但是实际上它非常频繁:IDE通常为我们生成代码,这就是为什么某些开发人员不考虑这一点的原因。 没有这个,几乎不可能对类的equals(Object other)
方法进行编码。
Java 9呢?
到目前为止,本文中没有关于Java 9的特定内容,而如今,每一篇Java文章都应该涉及Java 9(或已经是10)了。
如果我们一般地看一下访问控制,那么我们就不得不谈论JPMS,并且有很多关于此的文章。 codeFx上有很好的文章清单。 Stephen Colebourne的文章不错。
很快,您甚至可以从不同的出版商那里购买有关Java模块系统的书籍。 我很幸运,我已经可以阅读Packt的草稿,成为审稿人,我很喜欢。 但是,JPMS不会在此级别上更改“私有”。 仍然会有嵌套类,内部类和桥接方法,它们与以前完全相同。
几乎没有什么不同,Java 9现在在接口内部具有私有方法。 这意味着现在我们应该准备在内部和嵌套类中以及在接口中都具有合成桥方法。
起飞...
有时候最简单的事情并不像看起来那样简单。 毕竟,整个IT技术,科学和工程都不过是一堆零和一。 只是我们有很多。 真的很多 如果本文对您来说是新的内容,那么它应该告诉您Java语言和JVM中的某些领域您可能有兴趣研究更多。 例如:
- 嵌套类和内部类之间有什么区别?
- 您可以在类内部有一个嵌套接口,而在类内部也可以有一个内部接口?
- 接口中的类或接口呢? 接口中可以有内部类吗? 嵌套类怎么样?
- 您可以使用反射来编写列出类所有方法的代码吗? 会列出合成方法吗? 它会具有哪些修饰符?
- 编译内部类时,它将具有已编译名称
Outer$Inner.class
,这是一个合法名称。 但是,如果有Outer$Inner.java
源文件会发生什么? 想办法! - 生成的合成方法也具有合法名称。 如果使用该名称定义方法会怎样? 您看到的是Java规范还是特定于实现的?
- 您可以嵌套内部和嵌套的类和/或接口有多深? 嵌套类可以包含内部类吗? 内部类可以包含嵌套类吗?
- 您怎么猜,为什么JDK中没有合成修饰符的符号名? 为什么修改器的实际值与
volatile
字段的值相同? - 嵌套类中可以有一个静态字段,类或方法吗?
我知道,这些问题和知识的答案不切实际。 我从未见过任何代码或项目,这些代码或项目知道内部类不能具有静态字段会带来任何好处。 另一方面,考虑这些问题,获得答案可能会给您带来一些快乐,例如解决填字游戏(如果您喜欢的话)以及仍然有用的知识,以我们无法识别的方式帮助理解技术。 在某些情况下,一个人发现错误的速度比其他人快,因为她“感觉”到该技术。 那是当您无法说出是什么在耳边窃窃私语时,却是像上面那样的知识。 但是,只有当您喜欢深入研究该技术的那些细微之处时,才会如此。
最后一个技巧性的问题,如果您愿意,甚至比上面的娱乐问题还不如上面的实用:
难题
我们知道,内部(非嵌套)类内部不可能有静态字段。 仍然有可能由Java编译器从具有静态方法的内部类生成编译的类文件吗?
翻译自: https://www.javacodegeeks.com/2017/08/private-java-9.html