向上转型是父类引用指向子类对象,编译期按父类类型检查,运行期动态绑定方法调用;字段访问无多态性,子类特有方法不可见,需向下转型并校验类型。

父类引用指向子类对象就是向上转型,不是类型转换而是编译期与运行期的分工
它本质是 Java 多态的落地形式:声明用父类类型(编译期认这个),实际创建子类实例(运行期执行这个)。Animal a = new Dog() 这行代码里,a 只能调用 Animal 中声明过的方法,但若 Dog 重写了 eat(),运行时就会执行 Dog.eat() —— 这叫动态绑定,靠 JVM 的 invokevirtual 指令实现。
- 字段访问不具多态性:
a.name永远取Animal类定义的字段,哪怕Dog也声明了同名字段 - 子类特有方法不可见:
a.fly()编译直接报错,哪怕背后真是Bird实例 - 向上转型自动发生,无需
(Animal)强转;向下转型才需要显式强制,并伴随ClassCastException风险
为什么不能用父类引用直接调用子类新增方法
因为编译器只检查变量声明类型,不关心实际对象是谁。它要确保“这段代码在所有可能的子类下都安全”,所以只放行父类接口里明确定义的行为。
-
Animal a = new Bird(); a.fly();→ 编译失败:cannot resolve method 'fly()' - 想用
fly(),必须先向下转型:Bird b = (Bird) a;,但前提是a确实是Bird实例,否则运行时报ClassCastException - 更稳妥写法是加类型检查:
if (a instanceof Bird) { ((Bird) a).fly(); }
向上转型常见于哪些真实场景
不是为了炫技,而是为解耦和扩展留出空间。典型出现在集合统一管理、方法参数泛化、工厂返回值等地方。
- 集合存多种子类:
List<animal> animals = Arrays.asList(new Dog(), new Cat(), new Bird());</animal>后续统一animals.forEach(Animal::eat) - 方法参数接收任意子类:
void feed(Animal a) { a.eat(); },调用时传feed(new Dog())或feed(new Cat())都合法 - 工厂方法返回父类类型:
Animal create(String type) { return "dog".equals(type) ? new Dog() : new Cat(); },调用方无需 import 具体子类
容易忽略的关键细节:构造器中调用重写方法很危险
子类对象在构造过程中,父类构造器先执行,此时子类字段可能还没初始化,但若父类构造器里调用了被子类重写的方法,JVM 仍会跳转到子类版本——而这时子类状态是半成品。
立即学习“Java免费学习笔记(深入)”;
- 示例:
B b = new D();(D extends B),若B构造器中调func(),且D重写了它,func()内部访问D的字段可能为null或默认值 - 解决办法:避免在构造器中调用
protected或public方法;或把该逻辑移到init()等显式初始化方法中 - IDE 通常会警告 “calling overridable method in constructor” —— 别忽略它








