关于多态,首先理解以下四点内容
1、对象的静态类型:对象在声明时采用的类型。是在编译期确定的。
2、对象的动态类型:目前所指对象的类型。是在运行期决定的。
3、静态绑定(前期绑定):在程序执行前已经被绑定,对象的属性使用的是静态绑定。
4、动态绑定(后期绑定):在运行时根据具体对象的类型进行绑定,对象的方法一般都属于动态绑定。
(static和final修饰的方法为静态绑定)
(private修饰的方法会被编译器认为是final类型的,因为它无法被子类继承并重写)
对象的动态类型可以更改,但是静态类型无法更改;
举几个例子:
Father f = new Son();
含义:创建了一个Son对象,并把得到的引用立即赋值给Father
f的静态类型为Father,f的动态类型为Son;
此时的f无法访问Father中的私有属性和方法
Class Father{
name = "father";
}
Class Son extends Father{
name = "son";
age = 20;
}
由于属性是静态绑定的,f的静态类型为Father,所以f.name寻找的是Father中的name;
而age属性在Father中不存在,所以f.age将报错
Class Father{
public void a(){
system.out.print("father")
}
}
Class Son extends Father{
public void a(){
system.out.print("son")
}
}
由于方法是动态绑定的,f的动态类型为Son,所以f.a()将调用Son的方法
Class Father{
public static void a(){
system.out.print("father")
}
}
Class Son extends Father{
public static void a(){
system.out.print("son")
}
}
由于静态方法是静态绑定的,f的静态类型为Father,所以f.a()将调用Father的方法
最后再来看一个例子
Class Father{
private void a(){//私有方法
system.out.print("father")
}
}
Class Son extends Father{
public void a(){
system.out.print("son")
}
}
由于私有方法在编译器中会被默认为final类型,故为静态绑定,f的静态类型为Father,所以f.a()将调用Father的方法。(PS:此时Son中的a方法是一个全新的方法)