Java多态由继承、方法重写和父类引用指向子类对象三者自然达成,仅对实例方法重写生效,编译时看声明类型,运行时按实际类型动态绑定。

Java里多态不是靠某个“开关”或注解开启的,而是通过继承 + 方法重写 + 父类引用指向子类对象这三者自然达成的。缺一不可。
父类引用调用被重写的方法时才体现多态
多态只在运行时发生,且仅针对实例方法的重写(override),不适用于静态方法、私有方法、构造器或字段访问。
-
Animal a = new Dog(); a.speak();—— 若speak()在Dog中被重写,则执行Dog版本 -
a.name—— 访问的是Animal类中定义的字段,与实际对象类型无关 -
Animal.staticMethod()—— 编译期绑定,调用的是声明类型(Animal)的静态方法,无多态
接口实现也是多态的合法路径
除了类继承,实现接口同样构成多态:接口类型变量可引用任意实现类对象,调用的是具体实现类中重写的方法。
-
List是典型多态应用list = new ArrayList(); -
list.add("x")实际执行ArrayList.add(),但编译时只认List接口定义 - 切换为
new LinkedList()时,行为改变但调用方式完全不变
编译时类型决定能否调用,运行时类型决定执行哪个版本
这是最容易混淆的点:编译器只看引用变量的声明类型,而 JVM 在运行时根据真实对象类型查虚方法表(vtable)来分派方法。
立即学习“Java免费学习笔记(深入)”;
-
Object obj = new String("hello"); obj.toString();—— 编译通过,因为Object有toString() -
obj.length();—— 编译报错,Object没有length()方法,哪怕obj实际是String - 若强制转型:
((String) obj).length();才能调用,此时风险由开发者承担
真正难的不是写出来,而是判断某行代码是否发生了多态——得同时看清左边声明类型、右边创建类型、方法是否可被重写、有没有被 final 修饰,以及是否被 static/private 隐藏。这些细节一旦漏看,就容易误以为“应该多态”却没生效。








