多态的本质是通过虚方法表(vtable)实现运行时动态绑定,每个类有静态只读vtable,对象通过vptr指针访问,调用时按编译期确定的索引O(1)查表跳转;private、static、final/non-virtual方法不入vtable,接口多态则使用itable机制。
多态的本质,是让同一段调用代码,在运行时根据对象的真实类型,自动找到并执行对应版本的方法。它不靠猜测,也不靠运行时反复比对签名,而是依赖一张提前建好、静态固定的跳转表——虚方法表(vtable)。
vtable 是类级别的只读结构,编译或类加载时就生成完毕,存放在程序的只读数据段(如 .rodata),所有该类实例共享同一份表。而每个对象在内存布局最开头,都会隐式插入一个指针(vptr),指向自己所属类的 vtable。这个指针大小固定(32 位系统占 4 字节,64 位占 8 字节),是对象额外付出的唯一空间代价。
当通过指针或引用调用虚函数时,CPU 执行的是标准流程:
比如 Base* p = new Derived(); p->func();,实际执行的是“取 p 的 vptr → 查 vtable 第 N 项 → 跳过去运行”,N 在编译时就固化了,和 p 到底指向谁无关。
vtable 只收录参与动态绑定的方法。以下三类方法不会出现在表中,调用时直接走静态指令:
当调用发生在接口引用上(如 Runnable r = ...; r.run();),JVM 不查 vtable,而是查 itable。itable 按接口维度组织,每个实现类为每个所实现的接口单独维护一张映射表,解决接口方法到具体实现的定位问题。这和 vtable 的“继承链扁平化”思路不同,但目的相同:在运行时以最小开销命中正确实现。