性能文章>最新研发面试题,Java中的静态方法为什么不能调用非静态方法>

最新研发面试题,Java中的静态方法为什么不能调用非静态方法原创

2755310

哈喽,我是子牙。十余年技术生涯,一路披荆斩棘从技术小白到技术总监到JVM专家到创业。技术栈如汇编、C语言、C++、Windows内核、Linux内核。特别喜欢研究虚拟机底层实现,对JVM有深入研究。分享的文章偏硬核,很硬的那种。

 

手撸过JVM、内存池、垃圾回收算法、synchronized、线程池、NIO…

 

 

昨晚京东大佬勇哥在群里分享了一道他新创的JVM面试题,我听完后觉得还挺有意思的,分享给大家

 

 

小佬们先别急着看我的分析,先自己想想答案

 

 

 

你是不是想说

 

因为静态方法是属于类的,而非静态方法属于实例对象

 

 

哈,有人这样回答了,勇哥表示不满意。你这样回答,离正确答案,就差那么一句话。说出来了,就表示你知道了。哪句话呢?

 

 

临门一脚

 

JVM是如何调用Java方法的呢?咱们不把call_stub、entry_point、执行流扯进来,伪代码大概是这样:构建运行环境->调用Java方法。

 

构建运行环境做什么呢?比如是否需要触发类初始化、通过符号引用找到method对象,将method对象这个直接引用进行缓存、创建栈帧、拷贝参数……

 

不管是非静态方法,还是静态方法,JVM都是按这个流程进行调用。区别就在于构建运行环境这里。如果调用的是非静态方法,构建运行环境会多做一步:给this指针赋值。为了帮助大家理解这段话,我从我手写的JVM中把这段代码摘出来了。

 

 

这个就是问题的答案:

  1. 如果是静态方法调用非静态方法:因为调用静态方法时,是没有传入this指针的,所以在静态方法中调用非静态方法,非静态方法的第一个参数,是隐含的,无法传值,所以无法调用。其实你硬要支持也不是不可以做到,会稍微麻烦些,要改动很大机制才行。因为静态方法是所有Java对象共用的。但是如果真的支持了,那区分静态、非静态,就没意义了。

  2. 如果是非静态方法调用静态方法:可以调用。因为调用非静态方法,不需要this指针。其实就算要,也有的给。

 

GET到了吗?

 

超预期

 

如果我来面试,这个问题还没完,我会追问:其他的语言,比如C语言、C++、python、golang,你觉得它们会怎么做?为什么要追问呢?因为你就算回答出来上面的答案,其实也不能让你凸显出来。大概率很多人都能答出来。

 

如果我追问的,你能主动回答出来,那就非常nice。说明你对技术有立体的认知。但是很多Java Coder的技术是断层的。就是说他们的脑海中只有Java。你需要通过问题引导他思考。

 

python是怎么做的呢?调用python方法需要手动传this指针。因为不是隐含传参,所以这个问题放在python中问出,可能很容易答出来。

 

如果是C语言呢?看代码

 

 

没报错对不对?这不能说明在C语言中,静态方法能够调用非静态方法。在C语言中,static不是用来定义非静态方法的,是用来限制方法的作用域的。就是说,方法前面加了static修饰,这个方法只能在当前文件中调用。因为C语言是面向过程的语言。这里的方法称为函数更合适。

 

如果是C++呢?看代码

 

 

结果及背后的原理跟Java都是一样的:非静态方法能够调用静态方法,静态方法不能调用非静态方法,因为this指针。

 

 

推荐阅读

 

1、手写JVM小班到底教什么

2、困扰了你大半辈子的STW,今天总算可以毕业了

3、每个Coder必学!数据在计算机中是如何存储的

 

结语

 

我是子牙老师,一个明明可以靠脸吃饭,却偏偏抢大家饭碗的男人。

 

深入研究JVM、Linux内核、Windows内核,聚焦分享超硬核干货文章。还有,我的创业心得。运营公众号:硬核子牙。我的硬核文章及创业心得都在这里可以看到,目前原创文章已经很多很多啦。当然,还要我的手写JVM小班,如果你想进阶、想扎扎实实打下底层基础,欢迎跟我学习(jvm-ziya)

 

 

点赞收藏
分类:标签:
子牙_公号硬核子牙

对编程语言的设计与实现有浓厚兴趣。聚焦Hotspot源码、Linux内核研究,硬核干货分享

请先登录,查看3条精彩评论吧
快去登录吧,你将获得
  • 浏览更多精彩评论
  • 和开发者讨论交流,共同进步

为你推荐

一次 GDB 源码角度分析 jvm 无响应问题

一次 GDB 源码角度分析 jvm 无响应问题

10
3