多态不是语法糖,是运行时通过虚方法表动态绑定:编译看引用类型,运行看实际对象类型;static方法、字段、private/final方法不参与动态绑定。

多态不是语法糖,是运行时决定调用哪个方法
Java 中的多态本质是「编译期看引用类型,运行期看实际对象类型」。这意味着 javac 只检查引用类型是否声明了该方法,而真正执行哪段字节码,由 JVM 在运行时通过虚方法表(vtable)查表确定。如果你看到子类重写了父类方法但调用的还是父类实现,大概率是方法没被正确重写(比如参数类型不一致、static 修饰、private 或 final 修饰),或者根本没发生向上转型。
动态绑定只对实例方法生效,静态方法和字段不参与
以下情况不会触发动态绑定:
-
static方法:调用取决于引用类型,不是实际对象类型;Parent p = new Child(); p.staticMethod()调用的是Parent.staticMethod - 字段访问:
p.field读取的是Parent类中定义的字段值,哪怕Child里也定义了同名字段,也不会覆盖——这只是字段隐藏(hiding),不是重写(overriding) -
private/final/ 构造器方法:无法被重写,自然不进虚方法表
class Parent {
String name = "Parent";
static String type = "ParentStatic";
void say() { System.out.println("Parent.say"); }
static void speak() { System.out.println("Parent.speak"); }
}
class Child extends Parent {
String name = "Child";
static String type = "ChildStatic";
@Override
void say() { System.out.println("Child.say"); }
static void speak() { System.out.println("Child.speak"); }
}
Parent p = new Child();
System.out.println(p.name); // 输出 "Parent"(字段访问看引用类型)
System.out.println(p.type); // 输出 "ParentStatic"(静态字段同理)
p.say(); // 输出 "Child.say"(动态绑定生效)
p.speak(); // 输出 "Parent.speak"(静态方法不绑定)
方法签名必须严格一致,否则是重载而非重写
重写(override)要求方法名、参数列表(类型+顺序)、返回类型(协变返回除外)完全匹配。常见翻车点:
- 子类方法加了
throws Exception,但父类没声明——合法,但若抛出检查异常且未捕获,编译会报错 - 参数类型看似相同实则不同:比如父类是
Object,子类写成String—— 这是重载,不是重写,JVM 不会把它放进虚方法表对应位置 - 返回类型不满足协变规则:父类返回
Object,子类返回String合法;但父类返回String,子类返回Object编译失败
接口默认方法和私有方法不影响动态绑定逻辑
Java 8+ 接口可含 default 和 private 方法,但它们的绑定规则不变:
芒果系统GSHOP 纯静态商城系统,你还在为商城的优化而苦恼?GSHOP是全站纯静态商城系统,一键seo优化功能解决seo问题,自定义URL链接解决商城同质化问题;多页面显示:动态页、伪静态页面、纯静态页面增加收录,提升网站权重,提升流量等。安全稳定、功能强大的商城系统。1、芒果商城系统基于 php5.0开发,企业级应用。2、产品功能Ajax设计,响应速度更快,购物体验更好。3、全新密钥存放机制,
立即学习“Java免费学习笔记(深入)”;
-
default方法可被实现类重写,重写后参与动态绑定;未重写则走接口默认实现(此时实际调用链仍经由实现类的虚方法表跳转) -
private方法在接口内仅用于辅助其他default方法,不能被实现类继承或重写,也不进入绑定流程 - 当一个类同时继承父类并实现接口,且三者存在同签名方法时,优先级为:本类实现 > 父类方法 > 接口默认方法
容易忽略的是:即使接口提供了 default 方法,只要实现类没写 @Override,JVM 就不会把它当作重写目标——它只是“可用的默认实现”,不是“绑定候选”。









