父类引用指向子类对象
父类引用指向子类对象指的是:
例如父类Animal,子类Cat,Dog。其中Animal可以是类也可以是接口,Cat和Dog是继承或实现Animal的子类。
1 Animal animal = new Cat();
即声明的是父类,实际指向的是子类的一个对象。
这么使用的优点可以用这几个关键词来概括:多态、动态链接,向上转型
也有人说这是面向接口编程,可以降低程序的耦合性,即调用者不必关心调用的是哪个对象,只需要针对接口编程就可以了,被调用者对于调用者是完全透明的。让你更关注父类能做什么,而不去关心子类是具体怎么做的,你可以随时替换一个子类,也就是随时替换一个具体实现,而不用修改其他。以后结合设计模式(如工厂模式,代理模式)和反射机制可能有更深理解。
下面介绍java的多态性和其中的动态链接,向上转型:
面向对象的三个特征:封装、继承和多态;
- 封装隐藏了类的内部实现机制,可以在不影响使用者的前提下修改类的内部结构,同时保护了数据;
- 继承是为了重用父类代码,子类继承父类就拥有了父类的成员。
- 方法的重写、重载与动态链接构成多态性。Java之所以引入多态的概念,原因之一是它在类的继承问题上和C++不同,后者允许多继承,这确实给其带来的非常强大的功能,但是复杂的继承关系也给C++开发者带来了更大的麻烦,为了规避风险,Java只允许单继承,派生类与基类间有IS-A的关系(即“Cat”is a “Animal”)。这样做虽然保证了继承关系的简单明了,但是势必在功能上有很大的限制,所以,Java引入了多态性的概念以弥补这点的不足,此外,抽象类和接口也是解决单继承规定限制的重要手段。同时,多态也是面向对象编程的精髓所在。
-
理解多态,首先要知道“向上转型”:
我定义了一个子类Cat,它继承了Animal类,我可以通过
Cat c = new Cat();
实例化一个Cat的对象,这个不难理解。但当我这样定义时:
Animal a = new Cat();
这代表什么意思呢?
很简单,它表示我定义了一个Animal类型的引用,指向新建的Cat类型的对象。由于Cat是继承自它的父类Animal,所以Animal类型的引用是可以指向Cat类型的对象的。这就是“向上转型”。那么这样做有什么意义呢?因为子类是对父类的一个改进和扩充,所以一般子类在功能上较父类更强大,属性较父类更独特, 定义一个父类类型的引用指向一个子类的对象既可以使用子类强大的功能,又可以抽取父类的共性。 所以,父类类型的引用可以调用父类中定义的所有属性和方法,而对于子类中定义而父类中没有的方法,父类引用是无法调用的;
- 那什么是“动态链接”呢?当父类中的一个方法只有在父类中定义而在子类中没有重写的情况下,才可以被父类类型的引用调用; 对于父类中定义的方法,如果子类中重写了该方法,那么父类类型的引用将会调用子类中的这个方法,这就是动态链接。
下面看一下典型的多态例子:
1 class Father{ 2 public void func1(){ 3 func2(); 4 } 5 //这是父类中的func2()方法,因为下面的子类中重写了该方法 6 //所以在父类类型的引用中调用时,这个方法将不再有效 7 //取而代之的是将调用子类中重写的func2()方法 8 public void func2(){ 9 System.out.println("AAA"); 10 } 11 } 12 class Child extends Father{ 13 //func1(int i)是对func1()方法的一个重载,主要不是重写! 14 //由于在父类中没有定义这个方法,所以它不能被父类类型的引用调用 15 //所以在下面的main方法中child.func1(68)是不对的 16 public void func1(int i){ 17 System.out.println("BBB"); 18 } 19 //func2()重写了父类Father中的func2()方法 20 //如果父类类型的引用中调用了func2()方法,那么必然是子类中重写的这个方法 21 public void func2(){ 22 System.out.println("CCC"); 23 } 24 } 25 public class PolymorphismTest { 26 public static void main(String[] args) { 27 Father child = new Child(); 28 child.func1();//打印结果将会是什么? 29 child.func1(68); 30 } 31 }
上面的程序是个很典型的多态的例子。子类Child继承了父类Father,并重载了父类的func1()方法,重写了父类的func2()方法。重载后的func1(int i)和func1()不再是同一个方法,由于父类中没有func1(int i),那么,父类类型的引用child就不能调用func1(int i)方法。而子类重写了func2()方法,那么父类类型的引用child在调用该方法时将会调用子类中重写的func2()。
那么该程序将会打印出什么样的结果呢?
很显然,应该是“CCC”。
1 class Fu{ 2 int num=4; 3 void show1(){ 4 System.out.println("ShowFu"); 5 } 6 static void show2(){ 7 System.out.println("ShowFu"); 8 } 9 } 10 11 class Zi extends Fu{ 12 int num=5; 13 void show1(){ 14 System.out.println("ShowZi"); 15 } 16 static void show2(){ 17 System.out.println("ShowZi"); 18 } 19 } 20 21 public class Test { 22 23 public static void main(String[] args) { 24 Fu f = new Zi(); 25 Zi z = new Zi(); 26 System.out.println(f.num); //输出4 27 System.out.println(z.num); //输出5 28 f.show1(); //输出ShowZi 29 z.show1(); //输出ShowZi 30 f.show2(); //输出ShowFu (最好写成Fu.show2()) 31 z.show2(); //输出ShowZi (最好写成Zi.show2()) 32 } 33 34 }
多态执行:
成员变量、静态方法 编译运行都看左边
成员函数、非静态方法 编译看左边,运行看右边
个人分析:
把Fu f=new Zi();拆成两段:
1 Fu f; //声明一个Fu类的引用变量f,那么知道了f肯定是Fu类 2 f=new Zi(); //建立一个子类对象赋值给f
只有子类的函数覆盖了父类的函数这一个变化,但是f肯定是Fu这个类,也就是说f不可能变成其他比如Zi这个类等等(突然f拥有了Zi类特有函数,成员变量等都是不可能的)。所以f所代表的是函数被Zi类复写后(多态的意义)的一个Fu类,而Fu类原来有的成员变量(不是成员方法不可能被复写)没有任何变化。==>成员变量编译运行都看Fu。f的Fu类函数被复写了,==>非静态方法编译看Fu,运行看Zi
当方法静态时,Fu类的所有函数跟随Fu类加载而加载了。也就是Fu类的函数是先于对象建立之前就存在了,无法被后出现的Zi类对象所复写的,所以没发生复写,==>静态方法编译和运行都看Fu。
对于多态,可以总结以下几点:
一、使用父类类型的引用指向子类的对象;
二、该引用只能调用父类中定义的方法和变量;
三、如果子类中重写了父类中的一个方法,那么在调用这个方法的时候,将会调用子类中的这个方法;(动态连接、动态调用)
四、变量不能被重写(覆盖),”重写“的概念只针对方法,如果在子类中”重写“了父类中的变量,那么在编译时会报错。

浙公网安备 33010602011771号