编译期多态在编译时绑定方法,依赖重载,由方法名、参数类型和个数决定;运行期多态在运行时绑定方法,依赖重写和向上转型,由实际对象类型通过虚方法表动态确定。

编译期多态和运行期多态的本质区别在于:方法调用的绑定时机不同——前者在编译时就确定了具体调用哪个方法,后者则推迟到程序运行时才决定。
编译期多态:靠重载(Overload)实现
编译期多态主要体现为方法重载。Java编译器根据**方法名 + 参数类型 + 参数个数**(注意:不看返回值和修饰符)在编译阶段就选出唯一匹配的方法签名,并生成对应的字节码指令(如 invokestatic 或 invokevirtual,但目标已固定)。
- 重载方法必须在同一个类中,或子类对父类方法的重载(静态上下文)
- 编译器只看引用变量的声明类型,不关心实际对象是谁
- 例如:
print(String)和print(int)是两个独立方法,调用哪个由传入实参类型当场决定
运行期多态:靠重写(Override)+ 向上转型实现
运行期多态依赖继承、方法重写和父类引用指向子类对象(即向上转型)。JVM在运行时根据**实际对象的运行时类型**,通过虚方法表(vtable)动态查找并调用对应子类的重写版本,这个过程叫“动态绑定”或“晚绑定”。
- 必须有继承关系,子类重写父类的非private、非static、非final方法
- 调用方法时,编译看左边(引用类型),运行看右边(new出来的对象类型)
- 例如:
Animal a = new Dog(); a.sound();编译时认为是 Animal 的 sound,运行时执行 Dog 的 sound
关键区分点:看绑定发生在哪一步
一个简单判断方式:把代码写完后删掉某个子类,如果编译直接报错(比如找不到重载方法),那就是编译期多态;如果能编译通过,但运行时行为随对象不同而变化,那就是运行期多态。
立即学习“Java免费学习笔记(深入)”;
- 重载(Overload)→ 编译期绑定 → 静态分派
- 重写(Override)→ 运行期绑定 → 动态分派
- final、static、private 方法无法被重写,因此只能发生编译期绑定
基本上就这些。理解绑定时机,比死记“重载是编译时,重写是运行时”更有助于排查多态失效的问题。










