文章目录
- [一、什么是 Java 的多态](#一、什么是 Java 的多态)
- 二、多态成立的基本条件
- 三、方法重写与多态的关系
- 四、方法重载不是多态
- 五、成员变量是否具有多态性
- [六、static、final、private 方法与多态](#六、static、final、private 方法与多态)
- 七、接口与抽象类中的多态
- [八、JVM 层面:多态是如何实现的](#八、JVM 层面:多态是如何实现的)
- 九、多态在实际开发中的价值
- 十、总结
多态(Polymorphism)是 Java 面向对象编程中的核心特性之一,与封装、继承共同构成了面向对象设计的基础。理解多态不仅有助于写出更加灵活、可扩展的代码,也能帮助我们更好地理解 Java 框架(如 Spring)背后的设计思想。
一、什么是 Java 的多态
在 Java 中,多态指的是:
同一个父类(或接口)引用,在不同子类对象上调用同一个方法时,表现出不同的行为。
换句话说,方法的调用结果并不是由"引用的类型"决定,而是由"对象的实际类型"决定。
一个简单示例如下:
java
class Animal {
public void sound() {
System.out.println("animal");
}
}
class Dog extends Animal {
@Override
public void sound() {
System.out.println("wang");
}
}
class Cat extends Animal {
@Override
public void sound() {
System.out.println("miao");
}
}
public class Demo {
public static void main(String[] args) {
Animal a1 = new Dog();
Animal a2 = new Cat();
a1.sound();
a2.sound();
}
}
运行结果
csharp
wang
miao
变量看左边,方法看右边。
虽然变量类型都是 Animal,但在运行时会根据对象的真实类型调用不同的实现,这就是多态。
二、多态成立的基本条件
Java 中的运行时多态通常需要满足以下条件:
- 存在继承关系或接口实现关系
- 子类对父类方法进行了重写
- 使用父类或接口类型的引用指向子类对象
其中第三点通常也被称为"向上转型"。
java
Animal animal = new Dog();
这种写法是多态得以发挥作用的前提。
三、方法重写与多态的关系
多态依赖于方法重写(Override),而不是方法重载(Overload)。
方法重写要求:
- 方法名相同
- 参数列表相同
- 返回值类型兼容
- 访问权限不能变小
示例:
java
class Parent {
public void test() {
System.out.println("parent");
}
}
class Child extends Parent {
@Override
public void test() {
System.out.println("child");
}
}
当通过父类引用调用 test() 方法时,真正执行的是子类中的实现。
四、方法重载不是多态
虽然方法重载和多态在名字上容易混淆,但二者本质完全不同。
java
class Demo {
void test(int a) {}
void test(String s) {}
}
方法重载是在编译期根据参数类型决定调用哪个方法,不涉及运行时的动态绑定,因此不属于多态。
五、成员变量是否具有多态性
成员变量不具备多态性。
java
class A {
int x = 10;
}
class B extends A {
int x = 20;
}
A a = new B();
System.out.println(a.x);
输出结果为:
10
原因是:
成员变量的访问只与引用类型有关,与对象的实际类型无关。
常见总结为一句话:
变量看左边,方法看右边。
六、static、final、private 方法与多态
并不是所有方法都参与多态。
static 方法
static 方法属于类,而不是对象,在编译期就已经确定调用关系,因此不存在多态。
java
class A {
static void test() {
System.out.println("A");
}
}
class B extends A {
static void test() {
System.out.println("B");
}
}
A a = new B();
a.test(); // 输出 A
final 方法
final 方法不能被重写,因此也不具备多态。
private 方法
private 方法对子类不可见,无法被重写,也不存在多态行为。
七、接口与抽象类中的多态
接口多态
接口是 Java 中最常见、也是最重要的多态使用形式之一。
java
interface Payment {
void pay();
}
class Alipay implements Payment {
public void pay() {
System.out.println("alipay");
}
}
class WechatPay implements Payment {
public void pay() {
System.out.println("wechat");
}
}
Payment payment = new Alipay();
payment.pay();
通过接口类型引用不同实现类,可以在不修改调用方代码的情况下切换实现。
抽象类多态
抽象类同样可以作为多态的基础。
java
abstract class Shape {
abstract double area();
}
class Circle extends Shape {
double r;
Circle(double r) {
this.r = r;
}
double area() {
return Math.PI * r * r;
}
}
八、JVM 层面:多态是如何实现的
Java 的运行时多态在 JVM 层面主要依赖 虚方法表(Virtual Method Table,vtable)。
其核心机制是:
- 每个类在加载时都会生成一张虚方法表
- 表中存放的是可被重写的方法入口
- 方法调用时,根据对象的实际类型查找虚方法表
- JVM 使用
invokevirtual指令完成动态分派
以下方法不会进入虚方法表:
- static 方法
- final 方法
- private 方法
- 构造方法
九、多态在实际开发中的价值
多态的最大价值体现在以下几个方面:
- 降低模块之间的耦合
- 提高系统的可扩展性
- 支持面向接口编程
- 符合开闭原则(对扩展开放,对修改关闭)
例如:
java
List list = new ArrayList();
list = new LinkedList();
调用方代码无需修改,只需替换实现即可获得不同的行为。
十、总结
多态是 Java 面向对象设计中的核心机制,其本质是运行期动态绑定 。
方法的实际执行由对象的真实类型决定,而不是引用类型。
并非所有方法都具备多态特性,static、final、private 方法不参与多态。
接口和抽象类是实现多态的主要手段,多态也是现代 Java 框架设计的重要基础。