//学习笔记
多态
简单说:就是一个对象对应着不同类型。
多态在代码中的体现:父类或者接口的引用指向其子类的对象。
多态的好处:
提高可维护性(由多态前提所保证),提高了代码的扩展性
多态的弊端:
无法直接访问子类特有的成员。也就是说前期定义的内容不能使用(调用)后期子类的特有内容。
多态的前提:
1,子父类的继承关系
2,方法的重写
3,父类引用指向子类对象
另一个概念:动态绑定,指的是运行期间调用给的方法,是根据其具体的类型(只针对方法而言)
例子
abstract class Animal
{
abstract void eat();
}
class Dog extends Animal
{
void eat()
{
System.out.println("啃骨头");
}
void lookHome()
{
System.out.println("看家");
}
}
class Cat extends Animal
{
void eat()
{
System.out.println("吃鱼");
}
void catchMouse()
{
System.out.println("抓老鼠");
}
}
class Pig extends Animal
{
void eat()
{
System.out.println("饲料");
}
void gongDi()
{
System.out.println("拱地");
}
}
class DuoTaiDemo
{
public static void method(Animal a)
{
a.eat();
if(a instanceof Cat)//instanceof:用于判断对象的具体类型。只能用于引用数据类型判断
// //通常在向下转型前用于健壮性的判断。
{
Cat c = (Cat)a;
c.catchMouse();
}
else if(a instanceof Dog)
{
Dog d = (Dog)a;
d.lookHome();
}
else
{
}
}
public static void main(String[] args)
{
Animal a = new Cat(); //自动类型提升,猫对象提升了动物类型。
//但是特有功能无法访问。作用就是限制对特有功能的访问。
//专业讲:向上转型。将子类型隐藏。就不用使用子类的特有方法。
//如果还想用具体动物猫的特有功能。
//你可以将该对象进行向下转型。
// Cat c = (Cat)a; //向下转型的目的是为了使用子类中的特有方法。
// c.eat();
// c.catchMouse();
// 注意:对于转型,自始自终都是子类对象在做着类型的变化。
// Animal a1 = new Dog();
// Cat c1 = (Cat)a1;//
}
向上转型:将子类型隐藏,就不能使用子类的特有方法(方法的隐藏)。
向下转型:目的是为了使用子类中的特有方法。
多态时,成员的特点:
1,成员变量。
编译时:参考引用型变量所属的类中的是否有调用的成员变量,有,编译通过,没有,编译失败。
运行时:参考引用型变量所属的类中的是否有调用的成员变量,并运行该所属类中的成员变量。
简单说:编译和运行都参考等号的左边。
//举例
class Fu
{
int num = 3;
}
class Zi extends Fu
{
int num = 4;
}
class DuoTaiDemo3
{
public static void main(String[] args)
{
Fu f = new Zi();
System.out.println(f.num);
}
}
//输出的是3.
2,成员函数(非静态)。
编译时:参考引用型变量所属的类中的是否有调用的函数。有,编译通过,没有,编译失败。
运行时:参考的是对象所属的类中是否有调用的函数。
简单说:编译看左边,运行看右边。
因为成员函数存在覆盖特性。
//举例
class Fu
{
void show()
{
System.out.println("fu show");
}
}
class Zi extends Fu
{
void show()
{
System.out.println("zi show");
}
}
class DuoTaiDemo3
{
public static void main(String[] args)
{
Fu f = new Zi();
f.show();
}
}
//此时输出的是zi show
3,静态函数。
编译时:参考引用型变量所属的类中的是否有调用的静态方法。
运行时:参考引用型变量所属的类中的是否有调用的静态方法。
简单说,编译和运行都看左边。
其实对于静态方法,是不需要对象的。直接用类名调用即可。这也能解释为什么看的是左边了,因为左边才决定了该变量的类型。
//举例
class Fu
{
static void method()
{
System.out.println("fu static method");
}
}
class Zi extends Fu
{
static void method()
{
System.out.println("zi static method");
}
}
class DuoTaiDemo3
{
public static void main(String[] args)
{
Fu f = new Zi();
f.method();
}
}
//此时输出的是fu static method