1、多态是什么(也称动态绑定、后期绑定、运行时绑定)
绑定:将方法调用与方法主体关联起来称作绑定
前期绑定:在程序执行之前进行绑定
后期绑定(动态绑定、运行时绑定):在运行时根据对象的类型进行绑定,编译器一直不知道对象的类型,但是方法调用机制能找到正确的方法体,并加以调用。
注意:java中除了static方法和final方法(private方法属于final方法)外,其他所有的方法都是后期绑定。
class A{
public static void f(){
System.out.println("A.f()");
}
}
class B extends A{
public static void f() {
System.out.println("B.f()");
}
}
public class Test{
public static void main(String[] args){
g(new B());
}
public static void g(A a){
a.f();
}
}
//输出: A.() 静态方法不具有多态性
2、构造器内部的多态方法的行为
在基类构造器中调用动态绑定的方法,会调用导出类中被覆盖后的方法。(当然是使用导出类来创建对象实例时),但是在调用导出类中被覆盖的方法时,导出类的域可能还没初始化。例如:
class A{
public void f(){
System.out.println("A.f()");
}
public A(){
f();
}
}
class B extends A{
private int i=3;
public void f() {
System.out.println("B.f() "+i);
}
}
public class ReuseTest {
public static void main(String[] args) {
g(new B());
//创建B类对象实例,从输出可以看出A构造器调用的是B的f()方法,但是i还未被初始化为3
}
public static void g(A a) {
a.f();
}
}
//输出:
//B.f() 0
//B.f() 3
这是因为初始化的实际过程为:
(1)在其他任何事发生之前,将分配给对象的存储空间初始化为二进制的零(此时 i 被初始化为了0)
(2)然后按照继承的初始化方式进行初始化。
3、协变返回类型
Java SE5中添加了协变返回类型,他表示导出类中的被覆盖的方法可以返回基类方法的返回类型的导出类。
class A{
}
class B extends A{
}
class C{
public A f(){
}
}
class D extends C{
public B f(){ //此方法覆盖了C中的f()并且合法
}
}