方法重载发生在同一类中,靠参数列表(类型、数量、顺序)区分,是编译期行为;重写发生在子类对父类非private/static/final方法的重新实现,是运行期行为,遵循签名一致和协变返回规则。

方法重载发生在同一个类中,靠参数列表区分
重载(Overloading)是编译期行为,只看调用时传入的实参类型、个数和顺序。只要 方法名 相同但 参数列表 不同(类型、数量、顺序任一不同),就算重载,返回类型和访问修饰符不影响判断。
常见错误:以为改了 return 类型或 throws 异常就算重载——其实不会通过编译,JVM 会报 duplicate method 错误。
实操注意点:
-
int add(int a, int b)和double add(double a, double b)是合法重载 -
void print(String s)和void print(String... s)也是重载(可变参数本质是数组,签名不同) - 自动拆箱/装箱可能引发歧义,比如
method(int)和method(Integer)同时存在时,传5会优先匹配int版本
方法重写必须有继承关系,且遵循协变规则
重写(Overriding)是运行期行为,发生在子类对父类非 private、非 static、非 final 方法的重新实现。JVM 在运行时根据对象实际类型决定调用哪个版本。
立即学习“Java免费学习笔记(深入)”;
容易踩的坑:
- 子类方法签名必须与父类完全一致(包括方法名、参数列表),否则就是重载而非重写
-
@Override注解不是可选的——不加可能导致你以为重写了,其实只是定义了一个新方法(比如拼错方法名,或父类方法是protected而你写成public,但签名不符) - 返回类型可以是父类返回类型的子类型(协变返回),比如父类返回
Object,子类可返回String;但参数类型不能变窄或变宽 -
static方法不能被重写(只能被隐藏),哪怕子类写了同签名的static方法,调用仍取决于引用类型而非实际对象类型
重载解析在编译期完成,重写绑定在运行期发生
这是机制差异的核心。编译器看到 obj.method(1),会根据 obj 的声明类型(不是实际类型!)查找所有可用重载版本,选出最匹配的一个并固化为字节码中的符号引用;而 JVM 运行时才根据 obj 实际指向的对象类型,查虚方法表(vtable)跳转到真正的实现。
典型表现:
- 父类引用指向子类对象:
Animal a = new Dog();,若Animal有void speak(),Dog重写了它,则a.speak()调用的是Dog版本(重写生效) - 但如果
Animal有两个重载:void feed(String food)和void feed(Object o),而你写a.feed(new Dog()),编译器只看a的声明类型Animal,选的是feed(Object),哪怕运行时a指向Dog实例——重载选择与实际类型无关
构造器、私有方法、静态方法不参与重写,但可被重载
构造器天然只能在本类内调用,所以不存在“子类重写父类构造器”的概念;private 方法无法被继承,自然谈不上重写;static 方法属于类而非实例,绑定在类加载阶段,没有动态分派过程。
但它们都可以在同一个类里被重载:
-
public MyClass() {}和public MyClass(String name) {}是构造器重载 -
private void helper(int x)和private void helper(String s)是私有方法重载(仅本类可见) -
public static void log(String msg)和public static void log(String msg, Throwable t)是静态方法重载
真正容易混淆的是:子类定义了一个跟父类 static 方法同签名的 static 方法,这叫“隐藏”(hiding),不是重写——调用结果取决于变量声明类型,而不是对象实际类型。








