父类引用指向子类对象时无法直接调用子类特有方法,但可通过向下转型实现;例如Animal a = new Dog()可调用重写的eat()方法,体现多态,但调用bark()需先通过instanceof判断后强制转换为Dog类型,否则编译报错或运行时异常,因父类引用默认不可见子类新增成员,转型后才能访问。

父类引用 不能直接调用子类特有方法,但可以通过向下转型(downcasting)来实现间接调用。
父类引用指向子类对象的特性
在Java中,允许使用父类引用指向子类对象,这是多态的基础。例如:
class Animal { void eat() { System.out.println("Animal is eating"); } } class Dog extends Animal { void eat() { System.out.println("Dog is eating"); } void bark() { System.out.println("Dog is barking"); } } // 使用 Animal a = new Dog(); // 父类引用指向子类对象 a.eat(); // 调用的是Dog的eat方法,正常 a.bark(); // 编译错误!父类引用无法直接访问子类特有方法
通过向下转型调用子类方法
如果确定父类引用实际指向的是某个子类实例,可以将其强制转换为子类类型,然后调用子类方法:
Animal a = new Dog(); if (a instanceof Dog) { Dog d = (Dog) a; d.bark(); // 成功调用 }
说明:
立即学习“Java免费学习笔记(深入)”;
- instanceof 用于判断引用是否指向指定类的实例,避免 ClassCastException
- 向下转型需要显式强制转换语法 (子类类型)
- 运行时若类型不匹配会抛出异常,因此建议先做类型检查
只能调用子类重写的方法,不能访问子类新增成员
当父类引用调用一个被子类重写的方法时,实际执行的是子类的版本(动态绑定),这是多态的核心机制:
Animal a = new Dog(); a.eat(); // 输出 "Dog is barking",调用的是子类重写的方法
但即便如此,仍无法通过 a 访问 Dog 类独有的方法或字段,比如 bark()。
基本上就这些。父类引用本身视角受限,看不到子类扩展的部分,必须转型才能“还原”子类能力。


