java中多态是怎么一回事

Python047

java中多态是怎么一回事,第1张

多态字面上理解就是多种状态的意思,在java中就是同一个方法调用能获取不同的结果,体现在重写和重载上。其实是因为确定一个函数需要几个关键点:方法名,参数个数,参数类型,顺序,(返回类型在java中不能确定),(注意和参数名没有关系),那么同一个方法名就可以根据参数类型,参数个数,顺序不同写很多个方法,这时候我们调用同一个方法名,传入不同的参数就能返回不同的结果,也就是多态里面的重载了。另外重写是因为java当中有继承的概念,父类的部分方法(看权限修饰符)子类也能用,那么当子类中同样实现了这个方法,就是函数几个关键点要相同,包含返回类型,完全相同,那么当使用子类对象调用时,优先使用的是自己的方法,如果用父类的对象调用,那就是父类的方法,看具体的实现类,使用的场景很多,比如,java同一个接口有多个实现,当调用接口的方法时看具体的实现类。

在面向对象编程中,多态是指一个接口有多种实现方式,即一个接口使用不同实例执行不同的操作。在java中多态有两种表现形式:方法的重载和方法的重写。方法的重载是指同一个类中可以可以存在多个同名的方法,但这些方法的参数必须不同(个数,顺序,类型),程序运行时根据其参数的数量与类型调用对应的方法实现不同的功能。(与返回值类型没有关系)public

void

plant(int

i){方法体;}public

void

plant(String

s){方法体;}方法的重写是指子类提供与父类相同的方法,包括方法的名称,参数列表,返回值类型。 还有可以

父类

变量名=new

子类();

一、基本概念

多态性:发送消息给某个对象,让该对象自行决定响应何种行为。

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

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

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

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

二、Java多态性实现机制

SUN目前的JVM实现机制,类实例的引用就是指向一个句柄(handle)的指针,这个句柄是一对指针:

一个指针指向一张表格,实际上这个表格也有两个指针(一个指针指向一个包含了对象的方法表,另外一个指向类对象,表明该对象所属的类型);

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

The Java Virtual Machine does not require any particular internal structure for objects. In Sun 's current implementation of the Java Virtual Machine, a reference to a class instance is a pointer to a handle that is itself a pair of pointers: one to a table containing the methods of the object and a pointer to the Class object that represents the type of the object, and the other to the memory allocated from the Java heap for the object data. (jvm规范中关于对象内存布局的说明)

三、总结

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

DerivedC c2=new DerivedC()

BaseClass a1= c2//BaseClass 基类,DerivedC是继承自BaseClass的子类

a1.play() //play()在BaseClass,DerivedC中均有定义,即子类覆写了该方法

分析:

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

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

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

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

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

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

BaseClass a2=new BaseClass()

DerivedC c1=a2//出错

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

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

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

你可能说这个规则不对的,因为父类引用指向子类对象的时候,最后执行的是子类的方法的。

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

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

分析:

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

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