1. 多态
一个对象变量可以指示多种实际类型的现象。类型A的变量既可以引用一个A类型的对象,也可以引用A类型的任何一个子类的对象。
is-a规则——用来判断是否应将数据设计为继承关系
子类的每个对象也是超类的对象。
替换原则——is a的另一种表述
出现超类对象的任何地方都可以使用子类对象替换。比如,将子类的对象赋给超类变量。
2. 动态绑定
在运行时能够自动地选择适当的方法。(要调用的方法依赖于隐式参数的实际类型)
public class Parent{
public void say(){
System.out.println("parent said");
}
}
public class Child extends Parent{
public void say(){
System.out.println("child said");
}
public void eat(){
System.out.println("I am eating");
}
}
//test
Child c = new Child();
Parent[] ps = new parent[3];
ps[0] = c;
ps[1] = new Parent();
ps[2] = new Parent();
for(Parent p : ps){
p.say();
}
测试输出应为:
child said
parent said
parent said
虽然p声明为Parent类型,但实际上p既可以引用Parent类型的对象,也可以引用Child类型的对象。虚拟机知道p实际引用的对象类型,因此可以正确地调用相应的方法。当p引用Parent对象时,e.say()调用的是Parent类中的say()方法;当p引用Child对象时,e.say()调用的是Child类中的say()方法。
注意:ps[0]和c引用同一个对象,但编译器只将ps[0]看做一个Parent对象。所以,可以调用c.eat();
但不能调用ps[0].eat();
这是因为ps[0]声明的类型为Parent,但eat()不是Parent类的方法。
3. 覆盖(override)父类中的方法
在子类中定义一个与父类中方法签名相同的方法,那么子类中的这个方法就会覆盖父类中这个具有相同签名的方法。
注意1:方法签名——方法的名字和参数列表
注意2:保证返回类型的兼容性。允许子类将覆盖方法的返回类型改为原返回类型的子类型。
注意3:子类方法不能低于父类方法的可见性。父类方法为public,子类方法也必须声明为public。
4. 阻止继承——final类和方法
为了确保它们不会在子类中改变语义。
eg:
String类
Calendar类中的getTime、setTime方法
public final class Child extends Parent{
}
Child类——不允许扩展的类(不能派生Child类的子类),final类
public final void say(){
}
子类不能覆盖这个方法
- final字段,构造对象的时候必须被初始化,之后不能改变该字段的值