方法绑定分静态绑定和动态绑定:静态绑定在编译期根据变量声明类型确定,适用于private、static、final方法和构造器;动态绑定在运行时根据对象实际类型确定,是Java实现多态的核心,仅适用于实例方法。

方法绑定,就是把“调用某方法”这件事,和“到底执行哪段代码”连起来的过程。Java里不是所有方法都等运行时才决定调用谁——有些编译时就定死了,有些得看对象实际类型才能确定。关键不在“能不能变”,而在“什么时候定”。
静态绑定:编译期就锁死的调用
这类方法在写完代码、还没运行前,编译器就已经知道该调哪个实现。它只看变量声明的类型(比如 A a 中的 A),不关心 a 实际指向的是 A 还是它的子类对象。
- private 方法:不能被继承,自然只能调自己类里的版本
- static 方法:属于类,不是对象;子类“隐藏”而非“重写”,调用取决于引用类型
- final 方法:可继承但不可覆盖,子类调用的仍是父类那份代码
- 构造方法:每个类都有自己的构造逻辑,不参与多态,也不被继承
动态绑定:运行时按对象真实类型选方法
这是 Java 实现多态的核心机制。只要方法不是上面四类,JVM 就会在运行时查对象的实际类型(即 new 后面那个类),再从它的类信息中找对应方法。
- 调用的是对象的运行类型(如 A a = new B(),运行类型是 B)
- 查找顺序是:先在运行类型的类中找,没找到就沿继承链向上找,直到 Object
- 只对实例方法生效;属性、static 成员、构造器都不走这套规则
为什么属性没有动态绑定?
因为字段不支持重写(override),只有“隐藏”(hiding)。子类定义同名字段,只是新增一个独立变量,并不会替换父类字段。
立即学习“Java免费学习笔记(深入)”;
- A a = new B() 中,a.count 取的是 A 类里声明的 count 值
- 即使 B 也定义了 count = 20,它和 A.count 是两个不同内存位置的变量
- 想让值随对象类型变化?必须通过 getter 方法——方法才走动态绑定
一个典型例子看懂区别
假设 A 有 int i = 10 和 int getI() { return i; };B extends A 重写了 getI() 并定义 int i = 20:
- a.getI() → 调 B.getI() → 返回 20(动态绑定,走方法)
- a.i → 访问 A.i → 返回 10(静态绑定,看声明类型)
- ((B)a).i → 强转后访问 B.i → 才能得到 20
基本上就这些。理解绑定,本质是分清“编译时能看见什么”和“运行时实际是什么”。










