java多态性《下》

来源:互联网 发布:人工智能专业就业前景 编辑:程序博客网 时间:2024/04/29 20:56

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

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

分析:

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

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

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

jvm关于多态性支持解决方法是和c++中几乎一样的,

只是c++中编译器很多是把类型信息和虚拟函数信息都放在一个虚拟函数表中,但是利用某种技术来区别。

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

虚拟函数调用是经过虚拟函数表间接调用的,所以才得以实现多态的。

Java的所有函数,除了被声明为final的,都是用后期绑定。

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

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

class Human{

void run(){输出 人在跑}

}

class Man extends Human{

void run(){输出 男人在跑}

}

这个时候,同是跑,不同的对象,不一样(这个是方法覆盖的例子)

class Test{

void out(String str){输出 str}

void out(int i){输出 i}

}

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

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

Human ahuman=new Man();


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

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

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

这2句话,就是最好的证明,因为不知道它是大熊猫,但知道它的父类是动物,所以,

这个大熊猫对象,你把它当成其父类 动物看,这样子合情合理.

这种方式下要注意 new Man();的确实例化了Man对象,所以 ahuman.run()这个方法 输出的 是 "男人在跑 "

如果在子类 Man下你 写了一些它独有的方法 比如 eat(),而Human没有这个方法,

在调用eat方法时,一定要注意 强制类型转换 ((Man)ahuman).eat(),这样才可以...

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

实例:

package domatic;

//定义超类superA

class superA {

int i = 100;

void fun(int j) {

j = i;

System.out.println("This is superA");

}

}

// 定义superA的子类subB

class subB extends superA {

int m = 1;

void fun(int aa) {

System.out.println("This is subB");

}

}

// 定义superA的子类subC

class subC extends superA {

int n = 1;

void fun(int cc) {

System.out.println("This is subC");

}

}

class Test {

public static void main(String[] args) {

superA a = new superA();

subB b = new subB();

subC c = new subC();

a = b;

a.fun(100);

a = c;

a.fun(200);

}

}

/*

* 上述代码中subB和subC是超类superA的子类,我们在类Test中声明了3个引用变量a, b,

* c,通过将子类对象引用赋值给超类对象引用变量来实现动态方法调用。也许有人会问: * "为什么(1)和(2)不输出:This is superA"。

* java的这种机制遵循一个原则:当超类对象引用变量引用子类对象时,

* 被引用对象的类型而不是引用变量的类型决定了调用谁的成员方法,

* 但是这个被调用的方法必须是在超类中定义过的,

* 也就是说被子类覆盖的方法。

* 所以,不要被上例中(1)和(2)所迷惑,虽然写成a.fun(),但是由于(1)中的a被b赋值,

* 指向了子类subB的一个实例,因而(1)所调用的fun()实际上是子类subB的成员方法fun(), * 它覆盖了超类superA的成员方法fun();同样(2)调用的是子类subC的成员方法fun()。 * 另外,如果子类继承的超类是一个抽象类,虽然抽象类不能通过new操作符实例化,

* 但是可以创建抽象类的对象引用指向子类对象,以实现运行时多态性。具体的实现方法同上例。

* 不过,抽象类的子类必须覆盖实现超类中的所有的抽象方法,

* 否则子类必须被abstract修饰符修饰,当然也就不能被实例化了

*/

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


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

//父类

public class Father{

//父类有一个打孩子方法

public void hitChild(){

}

}

//子类1

public class Son1 extends Father{

//重写父类打孩子方法

public void hitChild(){

System.out.println("为什么打我?我做错什么了!");

}

}

//子类2

public class Son2 extends Father{

//重写父类打孩子方法

public void hitChild(){

System.out.println("我知道错了,别打了!");

}

}

//子类3

public class Son3 extends Father{

//重写父类打孩子方法

public void hitChild(){

System.out.println("我跑,你打不着!");

}

}

//测试类

public class Test{

public static void main(String args[]){

Father father;

father = new Son1();

father.hitChild();

father = new Son2();

father.hitChild();

father = new Son3();

father.hitChild();

}

}

都调用了相同的方法,出现了不同的结果!这就是多态的表现!

对于学习有困难不知道如何提升自己可以加扣:578024144进行交流得到帮助,获取学习资料




0 0
原创粉丝点击