多态
对象的多种形态
Student s = new Student()
person p = new Student()
把子类对象赋值给父类类型
对象可以根据父类类型的不同,灵活调用不同类中的方法
当一个方法需要能接收多个类时
可以将这个方法的参数设置为这多个类的父类
多态的前提
1,有继承或者实现关系
2,有父类引用指向子类对象
3,有方法重写
多态的好处
使用父类型作为参数,可以接收所有子类对象,体现多态的扩展和便利性
多态调用成员的特点
1,变量的调用
编译看左边,运行看左边
Animal a = new Dog();
sout(a.name)
如果父类中没有这个变量,会直接报错->编译看左边(父类)
运行出的结果会是父类中的变量结果->运行看左边(父类)
2,方法的调用
编译看左边,运行看右边
如果父类中没有这个方法,会直接编译失败->编译看左边(父类)
在运行代码时,实际上运行的是子类中的方法->运行看右边(子类)
(虚方法表重写后,会覆盖父类的方法)
从内存的角度理解
1,在加载字节码文件时,先加载父类,再加载子类
每个类的下面包含所有的成员变量和成员方法以及虚方法表
2,在创建对象时,会在堆内存中开辟两个空间
一个存储父类成员变量一个存储子类成员变量
会根据对象类型去寻找变量
如果是父类对象则先去父类空间找
如果是子类对象则先去子类空间找
如果没有找到就会直接编译错误
3,在对象调用方法时,会根据对象类型先去找父类字节码文件中的方法,如果没有直接编译错误
但实际运行时调用的是子类重写后的方法
多态的优势
在多态形式下,右边对象可以实现解耦合,便于扩展和维护
1,如果需要更改调用的方法,只需要更改new之后代码也就是创建对象的代码即可
利于维护
2,如果使用父类类型作为参数,可以接收所有子类参数
多态的便利性
eg:在集合添加泛型时,可以通过添加父类类型来增加集合内可添加的类型
多态的弊端
不能调用子类中的特有功能(父类中没有子类有的功能)
解决方案:可以用一个子类对象接收强制转换后的多态对象
可以用instanceof 判断类型
eg: if(a instanceof b)