登录
首页 >  文章 >  java教程

方法重载与重写的核心区别解析

时间:2026-04-13 10:57:58 354浏览 收藏

方法重载与重写虽都表现为“同名方法”,却是Java多态机制中截然不同的两个核心概念:重载是编译期的静态多态,发生在同一类内,靠参数列表差异区分,由编译器在编码时就决定调用哪个版本;而重写是运行期的动态多态,依托继承关系,要求方法签名完全一致,由JVM根据对象实际类型动态选择执行逻辑。二者在发生位置、绑定时机、签名约束、修饰符限制(如static/final/private的参与规则)等方面均存在根本性对立——理解这一区别,不仅是掌握面向对象设计的关键,更是避免运行时行为误判、写出可维护可扩展代码的基石。

Java 方法重载(Overload)与方法重写的本质区别

方法重载和方法重写看起来都是“同名方法”,但它们发生的场景、编译器处理方式和运行时行为完全不同——核心区别在于:重载是 编译期绑定(静态多态),解决的是“同一个类中多个同名方法如何选择”的问题;重写是 运行期绑定(动态多态),解决的是“子类如何改变父类已有行为”的问题。

发生位置与继承关系不同

重载必须发生在同一个类内部,与继承无关。只要参数列表不同(类型、个数、顺序),就可以在同一个类里定义多个同名方法。

重写必须发生在有继承关系的两个类之间(子类重写父类方法),且方法签名(名称 + 参数列表)必须完全一致,返回类型需协变或相同,访问权限不能更严格。

  • 重载示例:class A { void print(int x) {} void print(String s) {} } —— 同一个类,参数不同
  • 重写示例:class B extends A { @Override void print(int x) {} } —— 子类覆盖父类同签名方法

判断依据和编译器行为不同

重载的解析完全由编译器在编译阶段完成,依据是调用处的实参类型和个数。编译器会选出最匹配的那个重载版本,不关心对象实际类型。

重写的调用则由JVM在运行时根据对象的实际类型(而非引用类型)决定,即所谓“动态分派”。即使父类引用指向子类对象,也会执行子类重写后的方法。

  • 重载:A a = new B(); a.print(10); → 编译看 a 的声明类型 A,找 A 中参数为 int 的 print
  • 重写:A a = new B(); a.print(10); → 运行看 a 实际是 B,执行 B 中重写的 print

对方法签名的要求相反

重载要求方法名相同,但参数列表必须不同;返回类型、异常、修饰符可任意(甚至可以是 private 或 static)。

重写要求方法名和参数列表必须完全相同;返回类型需是父类返回类型的子类型(协变返回)、抛出的异常不能新增检查异常、访问修饰符不能比父类更严格(如父类 protected,子类不能是 private)。

  • 重载允许:void show() 和 int show(String) —— 名字同、参数不同、返回不同,合法
  • 重写不允许:父类 void calc(),子类 int calc() —— 返回类型不协变,编译报错

static、final、private 方法能否参与?

static 方法可以重载,但不能被重写(只能被“隐藏”,即子类定义同签名 static 方法,属于静态绑定,与多态无关)。

final 方法可以重载,但不能被重写(编译器直接禁止子类覆盖)。

private 方法可以重载(在本类内),但不能被重写(子类根本不可见,谈不上覆盖;子类中同名方法只是新定义,与父类无关)。

  • private void run() {} 在类中可被重载为 private void run(int x) {}
  • 但子类写 void run() {} 并不是重写,而是全新方法,也不受 final 或 private 约束影响

今天关于《方法重载与重写的核心区别解析》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于的内容请关注golang学习网公众号!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>