多态

面向对象的编程语言里,「多态」是一个至关重要的概念。我们常说,面向对象的本质,是方法与数据的绑定。那对于一个拥有继承关系的类之间,方法的绑定,是终是子类「重写」父类的方法,通过父类的引用指向子类的对象,实现运行时的多态。

说起来比较绕,我们先以仅次于Hello World 著名的 「动物 - 狗」代码来说明多态,然后再来分析在 JVM 层面,多态是怎样实现的。

package com.example.demo;public class Demo {    public static void main(String[] args) {        Animal a = new Animal();        a.say();        Dog d = new Dog();        d.say();        Animal ad = new Dog();        ad.say();    }}class Animal {    public void say() {        System.out.println("Animal say");    }    public void play() {        System.out.println("play...");    }}class Dog extends Animal {    public void say() {        System.out.println("Dog say");    }}

输出的内容对于习惯了面向对象的 Java 开发者来说都比较熟悉

Animal sayDog sayDog say

那虚拟机是怎样知道到底要调用 Animal 的 say 还是 Dog 的say呢?

咱们从字节码的层面来看一下。

 0 new #2  3 dup 4 invokespecial #3 > 7 astore_1 8 aload_1 9 invokevirtual #4 12 new #5 15 dup16 invokespecial #6 >19 astore_220 aload_221 invokevirtual #7 24 new #5 27 dup28 invokespecial #6 >31 astore_332 aload_333 invokevirtual #4 36 return

你发现没有,在字节码的第9行,和第33行,分别对应到 d.say() 和 ad.say() ,但指令内容其实是一样的。这就神奇了。

在这两个方法执行前,第8行和第32行,会有一个aload的操作,前面的文章里有介绍过(看看 JVM 是怎样消化字节码指令的 ~~),是把这两个对象的引用 压到栈顶,给后面的操作用。这两个对象,一般也被称为方法的接收者(Receiver),如果熟悉 Golang等语言的朋友,对这个概念也不陌生。

从9行和第33行看,无论是方法调用的字节码指令还是参数,都指向了常量池的第4项。都是一样的,但最终结果并不相同。这里的重点在于 invokevirtual 这个指令的多态指行查找过程,即根据对象的 vtable 在运行时定位方法。

啥是 vtable?

前面的内容提到指令执行时从栈顶获取当前方法的「接收者」,通过invokerirtual 来执行这个接者者对应的方法。 注意这里的 virtual,和C++的虚方法类似。这个咱们不提,只说Java 的。

对象都有一个自己的「方法表」,这个表里除了自己的方法,还有从父类继承来的方法,甚至重写的父类的方法。所以,对应于重写重载,体现在方法表里也有所区别。每个子类继承父类的时候,都将直接复制一份父类的方法表,而对于父类方法的重写,会直接更新方法表里相同顺序的这个方法。

而重载,本质上由于签名及参数的区别,是一个新的方法,在方法表里会是新增一个元素。

这里的这个方法表,就是咱们说的 vtable(Virtual Method Table),表里的每个方法,对应的是它的实际执行入口地址。如果没有重写,那父类和子类的地址是一样的,都指向父类的实现。

如果子类重写之后,子类方法表里的这个方法的地址就指向了自己实现的版本。

而我们上面字节码处观察到的,两个 invokevirtual 对应的常量池索引序号是一样的,这样实现对于变换实现类型时,查找方法表只需要换个对象,索引依旧相同。

观察

理解了方法表大概的原理,我们来解剖下,上咱们的JVM「显微镜」(Java虚拟机的显微镜 Serviceability Agent)。

为了便于 Attach 到 Java 进程,可以在代码里加下 latch 进行 awiat 阻塞,启动 SA 就能观察了。

选择 ClassBrowser

在 Class列表里就能找到咱们上面创建的对象。@ 符号后面是这个对象对应的内存地址。复制上Dog的地址,再从菜单里选择Inspector,

你看 _vtable_len: 7

这是告诉我们 vtable 长度是7,里面有7个方法。

实际上咱们在这个类里只重写了父类 Animal 的 say方法,其它的是从 Animal 继承来的 play方法,以及超类 Object 里的 5个方法,大概这个样子

JVM 在首次加载类的时候,会解析类内包含的方法,方法解析之后就会计算当前类 vtable的大小。

可能你会问,Object 类内不止5个方法,为什么只算5个呢?而且我们新增其它static、 final 这一类的方法呢?

这里 vtable 只计算非static final 的,全部计算完就得出了vtable_len这个值。

每个 Java 的 Class 在 JVM 内部都会有一个自己的instanceKlass, vtable就分配在这个的最后。

整个instanceKlass的大小,在64位系统里大小是 0x1b8,记住它,后面用的着。  所以咱们上面看到了Dog 类的内存地址,继续找就能看到他其它方法对应的内存地址。

在Windows -> console 里执行这个:

mem 0x7C0060DD0 7

这个值怎么来的呢?是从对象的内存地址开始,加上 instanceKlass的大小。

0x7C0060DD0   =  0x00000007c0060c18 + 0x1b8

由于我们有7个方法,所以顺序查找7个地址。

所以你应该也发现了,Java 里对应这种重写的方法,是在类加载的时候,才能知道具体对应的是哪个方法,因此也被称为动态绑定或者迟绑定。

总结起来,这里的 vtable,相当于你的工具清单,有什么能力都做了罗列,像钢铁侠的各项技能,每个功能指向具体的超能力,在我们代码里可以把它理解成一个数组,数组的每个元素指向一个方法地址。

感兴趣的话,你加个static 的方法自己找找,看看在不在这里面呢?毕竟static方法执行不是有 invokestatic 指令嘛。

看点别的

怎样计算一个Java对象大小?这儿有几种方法~

听保洁的老大爷讲Java的垃圾回收

俯瞰Java 虚拟机

在 Java 里创建一个对象要经过哪些步骤?

看看 JVM 是怎样消化字节码指令的 ~~

你写下的try-catch-finally,在JVM看来不过是...

理解了1+2的过程,就理解了Java虚拟机

问题诊断神器BTrace

Java七武器系列霸王枪 -- 线程状态分析 jstack

Tomcat的异步Servlet实现原理

监控诊断工具 Arthas 源码原理分析

Java虚拟机的显微镜 Serviceability Agent

怎样回答技术面试题?

怎样了解你的线程在干嘛?

更多常见问题,请关注公众号,在菜单「常见问题」中查看,也欢迎加我微信,一起交流。

源码|实战|成长|职场

这里是「Tomcat那些事儿」

请留下你的足迹

我们一起「终身成长」

java 获取内存地址_Java 的多态在 JVM 里原来是这样的相关推荐

  1. java查看内存地址_Java内存机制和内存地址

    问题一: String str1 = "abc"; String str2 = "abc"; System.out.println(str1==str2); / ...

  2. java 获取http地址_java如何获取当前时间,java如何获取ip地址

    展开全部 获取当前时间public static void main(String[] args) throws IOException { SimpleDateFormat format = new ...

  3. C++ 获取内存地址(取值运算符)

    c++ 获取内存地址使用取值运算符 : & 下面看下& 的使用 #include <iostream> using namespace std; int main() {s ...

  4. python获取内存地址上存储的值

    在python中,可以通过id()这个方法来获取对象的内存地址. 但是反过来,怎么获取内存地址上存储的值? 先看一段代码: from ctypes import string_at from sys ...

  5. java获取IP地址和MAC地址方式

    java获取IP地址和MAC地址方式 前端和后台都可以获取 ip获取方式比较简单,就不做分析了,主要说mac的获取 前端获取的方式与局限性:尝试好几种方式,这种是最易实现.最简单的方式,确实能获取该浏 ...

  6. java 内存接口_java中多态机制的内存解析、抽象类、接口

    1.多态机制的内存解析 1).多态的三要素 *类之间具有相互继承关系: *子类重写父类的方法: *父类引用指向子类对象: 2).为了更加细致深入的了解多态运行的机制,将下列代码的在内存中运行的简图绘制 ...

  7. Java中获取内存地址

    在Python中,可以通过id()方法来获取变量所指向的内存空间的内存空间 class Bookstore:def __init__(self,Bookname,price):self.Booknam ...

  8. java 读取内存地址结构体_Java并发系列之volatile

    讲到Java并发,多线程编程,一定避免不了对关键字volatile的了解,那么如何来认识volatile,从哪些方面来了解它会比较合适呢?个人认为,既然是多线程编程,那我们在平常的学习中,工作中,大部 ...

  9. java客户端mac地址_Java 获取客户端mac地址

    package com.alpha.test; import java.io.BufferedReader; import java.io.IOException; import java.io.In ...

最新文章

  1. 年轻,误把unix当linux
  2. pyspark AttributeError: 'NoneType' object has no attribute 'setCallSite'
  3. CCNA课堂练习一:路由器链路备份功能
  4. java helloworld代码_java学习应用篇|逃不掉的HelloWorld
  5. 网管型工业交换机如何创建网络冗余
  6. python string 方法,python字符串的方法与操作大全
  7. vue表单中批量导入功能_vue实战(11)——vue+element UI实现表格数据导出Excel功能
  8. JAVA中for循环写杨辉三角_java使用for循环输出杨辉三角
  9. linux下的常用时间函数总结
  10. excel中最常用的30个函数_94个Excel常用函数目录
  11. 前后端分离后的权限控制设计​方案
  12. 告诉你一个真实的全球化
  13. Win10 System进程占用硬盘100%,Microsoft IME 占用CPU高
  14. 泰坦尼克号数据挖掘项目实战——Task1 数据分析
  15. CSGO 绑定一键跳投
  16. JAVA之bootstrap01
  17. RP-VIO:面向动态环境的基于平面的鲁棒视惯融合里程计(IROS2021)
  18. 合泰lcd的c语言程序,合泰单片机HT66F70A LCD12864液晶屏驱动 汇编和c语言都有
  19. 乔布斯在斯坦福的演讲
  20. 峰值检波电路的作用和原理_最简单的峰值检波电路

热门文章

  1. 快手与央视达成合作:除夕晚8点上快手看春晚
  2. 英雄联盟S11总决赛EDG夺冠,网友:立下的flag可以兑现了
  3. 苹果也开始打价格战了
  4. 关于9999元的小米铁蛋,这里有一份官方解答
  5. 原价19万的美系插混,2年后落地只要11万~15万,微蓝6 PHEV为啥这么惨
  6. 最高法明确:未成年直播打赏可退还
  7. 2999元起!Redmi K40系列不止骁龙888:还将标配双扬声器
  8. 美团“共享单车变球场”项目落地四川阿坝
  9. 没理由不买它!小米今年最后一款旗舰发布:性价比真的高
  10. 微信公布朋友圈9月十大谣言 包括新型手机病毒出现等