编译期多态在编译时绑定方法,依赖重载,由方法名、参数类型和个数决定;运行期多态在运行时绑定方法,依赖重写和向上转型,由实际对象类型通过虚方法表动态确定。
编译期多态和运行期多态的本质区别在于:方法调用的绑定时机不同——前者在编译时就确定了具体调用哪个方法,后者则推迟到程序运行时才决定。
编译期多态主要体现为方法重载。Java编译器根据**方法名 + 参数类型 + 参数个数**(注意:不看返回值和修饰符)在编译阶段就选出唯一匹配的方法签名,并生成对应的字节码指令(如 invokestatic 或 invokevirtual,但目标已固定)。
print(String) 和 print(int) 是两个独立方法,调用哪个由传入实参类型当场决定运行期多态依赖继承、方法重写和父类引用指向子类对象(即向上转型)。JVM在运行时根据**实际对象的运行时类型**,通过虚方法表(vtable)动态查找并调用对应子类的重写版本,这个过程叫“动态绑定”或“晚绑定”。
Animal a = new Dog(); a.sound(); 编译时认为是 Animal 的 sound,运行时执行 Dog 的 sound一个简单判断方式:把代码写完后删掉某个子类,如果编译直接报错(比如找不到重载方法),那就是编译期多态;如果能编译通过,但运行时行为随对象不同而变化,那就是运行期多态。

基本上就这些。理解绑定时机,比死记“重载是编译时,重写是运行时”更有助于排查多态失效的问题。