Java基础之多态性

来自:http://blog.chinabyte.com/a/894990.html

多态性是通过:

   

1 接口和实现接口并覆盖接口中同一方法的几不同的类体现的

   

2 父类和继承父类并覆盖父类中同一方法的几个不同子类实现的.

   

一、基本概念

   

多态性:发送消息给某个对象,让该对象自行决定响应何种行为。通过将子类对象引用赋值给超类对象引用变量来实现动态方法调用。

   

java 的这种机制遵循一个原则:当超类对象引用变量引用子类对象时,被引用对象的类型而不是引用变量的类型决定了调用谁的成员方法,但是这个被调用的方法必须是在超类中定义过的,也就是说被子类覆盖的方法。

   

1. 如果a是类A的一个引用,那么,a可以指向类A的一个实例,或者说指向类A的一个子类。

   

2. 如果a是接口A的一个引用,那么,a必须指向实现了接口A的一个类的实例。

   

二、Java多态性实现机制 
     

   

SUN目前的JVM实现机制,类实例的引用就是指向一个句柄(handle)的指针,这个句柄是一对指针:
    一个指针指向一张表格,实际上这个表格也有两个指针(一个指针指向一个包含了对象的方法表,另外一个指向类对象,表明该对象所属的类型);

   

另一个指针指向一块从java堆中为分配出来内存空间。

   

三、总结

   

1、通过将子类对象引用赋值给超类对象引用变量来实现动态方法调用。
   

   

  1. DerivedC c2=new DerivedC();  
  2. BaseClass a1= c2; //BaseClass 基类,DerivedC是继承自BaseClass的子类  
  3. a1.play(); //play()在BaseClass,DerivedC中均有定义,即子类覆写了该方法     
 
 

 

   

分析:

   

1、为什么子类的类型的对象实例可以覆给超类引用?

   

自动实现向上转型。通过该语句,编译器自动将子类实例向上移动,成为通用类型BaseClass;

   

2、a.play()将执行子类还是父类定义的方法?

   

子类的。在运行时期,将根据a这个对象引用实际的类型来获取对应的方法。所以才有多态性。一个基类的对象引用,被赋予不同的子类对象引用,执行该方法时,将表现出不同的行为。

   

在a1=c2的时候,仍然是存在两个句柄,a1和c2,但是a1和c2拥有同一块数据内存块和不同的函数表。

   

2、不能把父类对象引用赋给子类对象引用变量

   

 

   

  1. BaseClass a2=new BaseClass();  
  2. DerivedC c1=a2;//出错     
 
 

 

   

在java里面,向上转型是自动进行的,但是向下转型却不是,需要我们自己定义强制进行。

   

 

   

  1. c1=(DerivedC)a2; 进行强制转化,也就是向下转型.     
 
 

 

   

3、记住一个很简单又很复杂的规则,一个类型引用只能引用引用类型自身含有的方法和变量。

   

你可能说这个规则不对的,因为父类引用指向子类对象的时候,最后执行的是子类的方法的。
    其实这并不矛盾,那是因为采用了后期绑定,动态运行的时候又根据型别去调用了子类的方法。而假若子类的这个方法在父类中并没有定义,则会出错。

   

例如,DerivedC类在继承BaseClass中定义的函数外,还增加了几个函数(例如 myFun())

   

分析:

   

当你使用父类引用指向子类的时候,其实jvm已经使用了编译器产生的类型信息调整转换了。

   

这里你可以这样理解,相当于把不是父类中含有的函数从虚拟函数表中设置为不可见的。注意有可能虚拟函数表中有些函数地址由于在子类中已经被改写了,所以对象虚拟函数表中虚拟函数项目地址已经被设置为子类中完成的方法体的地址了。

   

4、Java与C++多态性的比较

   

jvm关于多态性支持解决方法是和c++中几乎一样的,只是c++中编译器很多是把类型信息和虚拟函数信息都放在一个虚拟函数表中,但是利用某种技术来区别。

   

Java把类型信息和函数信息分开放。Java中在继承以后,子类会重新设置自己的虚拟函数表,这个虚拟函数表中的项目有由两部分组成。从父类继承的虚拟函数和子类自己的虚拟函数。

   

虚拟函数调用是经过虚拟函数表间接调用的,所以才得以实现多态的。Java的所有函数,除了被声明为final的,都是用后期绑定。

   

四. 1个行为,不同的对象,他们具体体现出来的方式不一样,

   

比如: 方法重载 overloading 以及 方法重写(覆盖)override

   

 

   

  1. class Human{  
  2. void run(){输出 人在跑}  
  3. }  
  4. class Man extends Human{  
  5. void run(){输出 男人在跑}  
  6. }  
  7. 这个时候,同是跑,不同的对象,不一样(这个是方法覆盖的例子)  
  8. class Test{  
  9. void out(String str){输出 str}  
  10. void out(int i){输出 i}  
  11.    
 
 

 

   

这个例子是方法重载,方法名相同,参数表不同

   

ok,明白了这些还不够,还用人在跑举例

   

 

   

  1. Human ahuman=new Man();     
 
 

 

   

这样我等于实例化了一个Man的对象,并声明了一个Human的引用,让它去指向Man这个对象

   

意思是说,把 Man这个对象当 Human看了.

   

比如去动物园,你看见了一个动物,不知道它是什么, "这是什么动物? " "这是大熊猫! "

   

这2句话,就是最好的证明,因为不知道它是大熊猫,但知道它的父类是动物,所以,这个大熊猫对象,你把它当成其父类 动物看,这样子合情合理.这种方式下要注意 new Man();的确实例化了Man对象,所以 ahuman.run()这个方法 输出的 是 "男人在跑 "如果在子类 Man下你 写了一些它独有的方法 比如 eat(),而Human没有这个方法,在调用eat方法时,一定要注意 强制类型转换 ((Man)ahuman).eat(),这样才可以...

   

对接口来说,情况是类似的...

   

实例:

   

 

   

  1. package domatic;  
  2. //定义超类superA  
  3. class superA {  
  4. int i = 100;  
  5. void fun(int j) {  
  6. j = i;  
  7. System.out.println("This is superA");  
  8. }  
  9. }  
  10. // 定义superA的子类subB  
  11. class subB extends superA {  
  12. int m = 1;  
  13. void fun(int aa) {  
  14. System.out.println("This is subB");  
  15. }  
  16. }  
  17. // 定义superA的子类subC  
  18. class subC extends superA {  
  19. int n = 1;  
  20. void fun(int cc) {  
  21. System.out.println("This is subC");  
  22. }  
  23. }  
  24. class Test {   
  25. public static void main(String[] args) {  
  26. superA a = new superA();  
  27. subB b = new subB();  
  28. subC c = new subC();  
  29. a = b;  
  30. a.fun(100);  
  31. a = c;  
  32. a.fun(200);  
  33. }  
  34. }  
  35. /*  
  36. * 上述代码中subB和subC是超类superA的子类,我们在类Test中声明了3个引用变量a, b,  
  37. * c,通过将子类对象引用赋值给超类对象引用变量来实现动态方法调用。也许有人会问:  
  38. * "为什么(1)和(2)不输出:This is superA"。  
  39. * java的这种机制遵循一个原则:当超类对象引用变量引用子类对象时,  
  40. * 被引用对象的类型而不是引用变量的类型决定了调用谁的成员方法,  
  41. * 但是这个被调用的方法必须是在超类中定义过的,  
  42. * 也就是说被子类覆盖的方法。  
  43. * 所以,不要被上例中(1)和(2)所迷惑,虽然写成a.fun(),但是由于(1)中的a被b赋值,  
  44. * 指向了子类subB的一个实例,因而(1)所调用的fun()实际上是子类subB的成员方法fun(),  
  45. * 它覆盖了超类superA的成员方法fun();同样(2)调用的是子类subC的成员方法fun()。  
  46. * 另外,如果子类继承的超类是一个抽象类,虽然抽象类不能通过new操作符实例化,  
  47. * 但是可以创建抽象类的对象引用指向子类对象,以实现运行时多态性。具体的实现方法同上例。  
  48. * 不过,抽象类的子类必须覆盖实现超类中的所有的抽象方法,  
  49. * 否则子类必须被abstract修饰符修饰,当然也就不能被实例化了  
  50. */     
 
 

 

   

以上大多数是以子类覆盖父类的方法实现多态.下面是另一种实现多态的方法-----------重写父类方法

   

1.JAVA里没有多继承,一个类之能有一个父类。而继承的表现就是多态。一个父类可以有多个子类,而在子类里可以重写父类的方法(例如方法print()),这样每个子类里重写的代码不一样,自然表现形式就不一样。这样用父类的变量去引用不同的子类,在调用这个相同的方法print()的时候得到的结果和表现形式就不一样了,这就是多态,相同的消息(也就是调用相同的方法)会有不同的结果。举例说明:

   

 

   

  1. //父类  
  2. public class Father{  
  3. //父类有一个打孩子方法  
  4. public void hitChild(){  
  5. }  
  6. }  
  7. //子类1  
  8. public class Son1 extends Father{  
  9. //重写父类打孩子方法  
  10. public void hitChild(){  
  11. System.out.println("为什么打我?我做错什么了!");  
  12. }  
  13. }  
  14. //子类2  
  15. public class Son2 extends Father{  
  16. //重写父类打孩子方法  
  17. public void hitChild(){  
  18. System.out.println("我知道错了,别打了!");  
  19. }  
  20. }  
  21. //子类3  
  22. public class Son3 extends Father{  
  23. //重写父类打孩子方法  
  24. public void hitChild(){  
  25. System.out.println("我跑,你打不着!");  
  26. }  
  27. }  
  28. //测试类  
  29. public class Test{  
  30. public static void main(String args[]){  
  31. Father father;  
  32. father = new Son1();  
  33. father.hitChild();  
  34. father = new Son2();  
  35. father.hitChild();  
  36. father = new Son3();  
  37. father.hitChild();  
  38. }  
  39.  
©️2020 CSDN 皮肤主题: 大白 设计师:CSDN官方博客 返回首页