这篇文章主要探究了Java运行时多态中,子类重写了父类方法,并定义了和父类相同名称的成员变量,调用父类的方法,成员变量、静态变量等属性时,会具体调用子类还是父类的属性
先写一段Java多态的代码
父类
public class Animal {//父类
static int cons = 1;
int val = 1;
final int fin = 1;
public void eat() {
System.out.println("animal eat");
}
static void eat2() {
System.out.println("animal eat2");
}
}
子类
public class Cat extends Animal{
static int cons = 2;
int val = 2;
final int fin = 2;
@Override
public void eat() {
System.out.println("cat eat");
}
static void eat2() {
System.out.println("cat eat2");
}
}
调用测试
public static void main(String[] args) {
Animal a = new Cat();//父类引用
System.out.println(a.val);//成员变量
System.out.println(a.cons);//静态变量
System.out.println(a.fin);//常量
a.eat();//成员方法
a.eat2();//静态方法
}
根据多态的必要特点,猜一猜上述代码会输出什么?
继承
子类重写(override)父类方法
父类引用指向子类对象
下面是输出结果
1
1
1
cat eat
animal eat2
结论:一般可以浅显的认为多态中调用的属性是父类的,普通方法调用的是子类的,但我们这里可以用是否满足多态的三个特点来判断:
成员变量:通过多态调用的成员变量,无论是普通类型,静态类型,常量类型仍是父类的成员变量,因为成员变量不存在override(覆盖)问题
成员方法:通过多态调用的成员方法一般仍是父类中的方法,若在相应子类中被override(覆盖),则调用子类重写的方法。
静态方法:通过多态调用的静态方法仍是父类的静态方法,因为静态方法虽然能被子类的静态方法“重写”,但严格来讲算不上是override(覆盖)。
编译时多态:主要是方法的重载,通过参数列表的不同来区分不同的方法。
运行时多态:也叫作动态绑定,一般是指在执行期间(非编译期间)判断引用对象的实际类型,根据实际类型判断并调用相应的属性和方法。主要用于继承父类和实现接口时,父类引用指向子类对象。