父类应用指向子类对象指的是:
父类Animal,子类 Cat,Dog。当中Animal能够是接口或类,Cat和Dog是继承或实现Animal的子类。
Animal animal= new Cat();
声明的为父类,实际指向的是子类对象。我们先从内存角度理解。
如果Aninal父类,它里面的变量须要占用1M,它的子类Dog和Cat,须要占用0.5M内存。
通过代码看内存分配:
Animal animal=newAnimal();//系统将分配内存1M
Catcat = new Cat();系统将分布1.5M内存!由于子类有一个隐藏引用super会指向父类实例。所以实例化子类前会先实例化一个父类。先运行父类构造函数。由于包括了父类的实例,所以s能够调用父类的方法。
Cat cat1= cat,指向那1.5M内存.
Animal animalOne=(Animal)cat;//这时候animalOne会指向1.5M内存中1M内存,animalOne仅仅是指向cat中实例的父类实例对象。所以animalOne职能调用父类的方法,不能调用子类的方法(存储在0.5M内存中). Cat cat2=(Cat)animal//执行时会报ClassCatException。由于animal中仅仅有1M内存,而子类的引用都必需要有1.5M内存,所以无法转换。
Cat cat3=(Cat)animalOne;//这句能够通过执行,这时s3指向那1.5M的内存.因为f1是由s转换过来的,所以它是有1.5M的内存的,仅仅是它指向的仅仅有1M内存
以上是从内存的角度进行分析的。
从对象的角度看问题
我们先看几个关键词:多态,动态链接,向上转型
封装:隐藏了类的内部实现机制,能够在不影响使用者的前提下改动类的内部结构,同一时候保护了数据;
继承:是为了重用父类代码,子类继承父类就拥有了父类的成员。
方法的重写、重载与动态连接构成多态性。理解多态,首先要知道“向上转型”(子类转为父类)。
定义了一个子类Cat,它继承了Animal类,那么后者就是前者是父类。能够通过Catc = new Cat(); 实例化一个Cat的对象,这个不难理解。但当我这样定义时:Animal a = new Cat(); 这代表什么意思呢?它表示我定义了一个Animal类型的引用,指向新建的Cat类型的对象。因为Cat是继承自它的父类Animal,所以Animal类型的引用是能够指向Cat类型的对象的。这就是“向上转型”。
那么这样做有什么意义呢?
由于子类是对父类的一个改进和扩充,所以一般子类在功能上较父类更强大,属性较父类更独特,定义一个父类类型的引用指向一个子类的对象既能够使用子类强大的功能,又能够抽取父类的共性。所以,父类类型的引用能够调用父类中定义的全部属性和方法,而对于子类中定义而父类中没有的方法,父类引用是无法调用的;
◆那什么是动态链接呢?
当父类方法子类未重写,调用父类;若子类重写父类方法,调用子类方法,这就是动态连接。
class Father{ Public void func1(){ Func2(); } Public void fun2(){ System.out.println("AAA"); } } Class Child extends Father{ Public void func1(int i){ System.out.println("BBB"); } Public void func2(){ System.out.println("CCC"); } } Public class Test{ Public static void mian(String [ ]args){ Father child= new Child(); child.func1();//打印结果将会是什么? child.func1(89);//?该方法会如何呢? } }
class Child extends Father{ //func1(int i)是对func1()方法的一个重载,主要不是重写! public void func1(int i){ System.out.println("BBB"); } //func2()重写了父类Father中的func2()方法 //假设父类类型的引用中调用了func2()方法,那么必定是子类中重写的这种方法 public void func2(){ System.out.println("CCC"); } } public class PolymorphismTest { public static void main(String[] args) { Father child = new Child(); child.func1();//打印结果将会是什么? child.func1(68); //因为在父类中未定义这种方法,所以它不能被父类类型的引用调用 //所以在child.func1(68)会跑出异常的 } }
上面的程序是个非常典型的多态的样例。子类Child继承了父类Father,并重载了父类的func1()方法,重写了父类的func2()方法。重载后的func1(inti)和func1()不再是同一个方法,因为父类中没有func1(int i),那么,父类类型的引用child就不能调用func1(inti)方法。而子类重写了func2()方法,那么父类类型的引用child在调用该方法时将会调用子类中重写的func2()。
那么该程序将会打印出什么样的结果呢?非常显然,应该是“CCC”。
小结:
通过以上两种方式我们认识了父类引用指向子类,事实上是多态的应用,体如今下面几点
- 使用父类类型的引用指向子类的对象
- 该引用仅仅能调用父类中定义的方法和变量
- 假设子类中重写了父类中的一个方法,那么在调用这种方法的时候,将会调用子类中的这种方法;(动态连接、动态调用)
- 变量不能被重写(覆盖),”重写“的概念仅仅针对方法,假设在子类中”重写“了父类中的变量,那么在编译时会报错。