静态方法不能被重写,只能被隐藏;其调用由编译期根据引用声明类型决定,不参与多态,@Override注解对其无效。

静态方法不能被重写(override),只能被隐藏(hide)。
静态方法调用走编译期绑定
Java 中静态方法的调用由编译器在编译时决定,依据的是引用变量的声明类型,而不是运行时实际对象的类型。这和实例方法的动态绑定(virtual method invocation)完全不同。
-
static方法不参与多态,JVM 不会根据实际对象类型去查找子类中同签名的方法 - 即使子类定义了与父类完全相同的
static方法(相同方法名、参数列表、返回类型),也不会触发重写机制 - 这种“同名同参”的静态方法关系叫「方法隐藏」,不是重写;子类方法只是覆盖了父类方法在该类作用域下的可见性
重写 vs 隐藏:看调用方式就清楚
下面这个例子能直接看出区别:
class Parent {
static void show() {
System.out.println("Parent.show()");
}
void display() {
System.out.println("Parent.display()");
}
}
class Child extends Parent {
static void show() {
System.out.println("Child.show()");
}
@Override
void display() {
System.out.println("Child.display()");
}
}
public class Test {
public static void main(String[] args) {
Parent p = new Child();
p.show(); // 输出:Parent.show() ← 编译看 p 的声明类型(Parent)
p.display(); // 输出:Child.display() ← 运行看 p 实际类型(Child)
Child c = new Child();
c.show(); // 输出:Child.show() ← 声明类型是 Child,所以调用 Child 的静态方法
}
}
关键点:
立即学习“Java免费学习笔记(深入)”;
-
p.show()调用的是Parent.show(),因为p是Parent类型引用 -
p.display()调用的是Child.display(),因为display()是实例方法,走虚方法表查找 - 编译器对
show()的解析发生在编译期,不会等到运行时再查子类
@Override 注解对静态方法无效
如果你试图在子类的静态方法上加 @Override,编译器会报错:
error: method does not override or implement a method from a supertype
这是因为 @Override 的语义只适用于可被重写的方法(即非 static、非 private、非 final 的实例方法)。给静态方法加这个注解,说明你误以为它参与继承多态,是个典型的设计误解。
- 加了
@Override→ 编译失败 → 提醒你这不是重写 - 没加但写了同签名静态方法 → 编译通过 → 但属于隐藏,不是多态行为
- 如果父类方法是
private static,子类写同名方法更不是隐藏,而是完全无关的新方法(连继承关系都没有)
真正容易被忽略的是:很多开发者在重构时把一个原本是实例方法的工具逻辑改成 static,却没意识到调用点上的多态语义彻底消失了——尤其是当该方法被父类引用调用时,行为会静默退化为“按声明类型调用”,而不是“按实际类型分发”。








