java中的多态性 是怎么体现的呢?

min3278 2007-03-05 11:52:48
一面试题,我答的不是很全。大家一起讨论一下。
...全文
3689 15 打赏 收藏 转发到动态 举报
写回复
用AI写文章
15 条回复
切换为时间正序
请发表友善的回复…
发表回复
蒙古狼-mgl 2012-06-08
  • 打赏
  • 举报
回复
dangerrei回答的非常棒
atomic_age总结的很好,例子也给的很好。

多态:父类和接口的有了统一的写法的机制(通过extends和implements形成的多个类的实例化),而通过接口和父类可以生出了多个儿子(类和子类,或子子类),input一致但output却不同。

overload:中文叫重载。在同一类中,存在多个同名方法,但方法的参数不同。(不含返回值,专指参数个数,参数类型不同)

override:中文叫覆盖。就是他本身的意思,如类覆盖抽象类的抽象方法方法,如类覆盖接口定义的方法等,要求很严格,要求三要素一致(如c语言的函数声明一样),原来的方法并没有放弃。

overwrite:中文叫重写。也是它本身的意思,如子类重写父类的方法。重写之后,子类只能使用自己的方法,原来父类的方法不可用了。

多态是一种设计思想,而overload,overwrite和override是一个实现方式。如多态的方法实现是靠override来实现的。
ghlong 2011-09-19
  • 打赏
  • 举报
回复
[Quote=引用 11 楼 zjunzhang 的回复:]

呵呵,上面我说错了!
多态是通过:
1 接口 和 实现接口并覆盖接口中同一方法的几不同的类体现的
2 父类 和 继承父类并覆盖父类中同一方法的几个不同子类实现的.
[/Quote]++
wind__dance 2008-07-27
  • 打赏
  • 举报
回复
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修饰符修饰,当然也就不能被实例化了
*/
GProgrammer 2008-07-26
  • 打赏
  • 举报
回复
[Quote=引用 10 楼 atomic_age 的回复:]
interface Parent
{
String method();
}

class Child1 implements Parent
{
public String method()
{
return "Child1 ";
}
}

class Child2 implements Parent
{
public String method()
{
return "Child2 ";
}
}

public class Test
{
public static void main(String[] args)
{
Parent parent = new Child1();
System.out.println(parent.meth…
[/Quote]

很清楚,很明了。
Buffered 2007-03-07
  • 打赏
  • 举报
回复
是不是方法的 覆盖或是 方法的 重写啊?
luyang1016 2007-03-07
  • 打赏
  • 举报
回复
现在有个比较流行的多态问题,lz可以研究一下

package org.luyang.csdn;

public class Parent {
public void test() {
}

public Parent() {
test();
}

public static void main(String[] args) {
new Child();
}
}


class Child extends Parent {
private int instanceValue = 20;

public Child() {
}
public void test() {
System.out.println("instance value is: " + instanceValue);
}
}


请问打印出来的结果是什么?
朝梦夕拾 2007-03-07
  • 打赏
  • 举报
回复
呵呵,上面我说错了!
多态是通过:
1 接口 和 实现接口并覆盖接口中同一方法的几不同的类体现的
2 父类 和 继承父类并覆盖父类中同一方法的几个不同子类实现的.
atomic_age 2007-03-07
  • 打赏
  • 举报
回复
interface Parent
{
String method();
}

class Child1 implements Parent
{
public String method()
{
return "Child1";
}
}

class Child2 implements Parent
{
public String method()
{
return "Child2";
}
}

public class Test
{
public static void main(String[] args)
{
Parent parent = new Child1();
System.out.println(parent.method());
parent = new Child2();
System.out.println(parent.method());
}
}

输出结果:
Child1
Child2


只有多个子类从一个父类继承或实现一个接口。 在建立这些子类实例时,都用父类或接口做为变量类型,如上例中的parent。也就是说,用户对应的接口都是一个Parent。而由于new后面的子类不同,而产生调用同一个方法method返回不同结果的显现叫多态。就是同一个方法在使用不同子类时有不同的表现(在这里是不同的返回值)。

方法重载不叫多态。
朝梦夕拾 2007-03-06
  • 打赏
  • 举报
回复
通过多个同名方法中的参数的不同体现(不包括顺序不同)
min3278 2007-03-06
  • 打赏
  • 举报
回复
方法的重载和类的重写,我也是这样回答的,然后面试的那人个说我说的不全面.知道面比较窄,,,我晕哦.
左大神在这 2007-03-05
  • 打赏
  • 举报
回复
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(),这样才可以...

对接口来说,情况是类似的...
左大神在这 2007-03-05
  • 打赏
  • 举报
回复
重载和重写.
dangerrei 2007-03-05
  • 打赏
  • 举报
回复
一、基本概念

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

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已经使用了编译器产生的类型信息调整转换了。
这里你可以这样理解,相当于把不是父类中含有的函数从虚拟函数表中设置为不可见的。注意有可能虚拟函数表中有些函数地址由于在子类中已经被改写了,所以对象虚拟函数表中虚拟函数项目地址已经被设置为子类中完成的方法体的地址了。


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

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

Java把类型信息和函数信息分开放。Java中在继承以后,子类会重新设置自己的虚拟函数表,这个虚拟函数表中的项目有由两部分组成。从父类继承的虚拟函数和子类自己的虚拟函数。
虚拟函数调用是经过虚拟函数表间接调用的,所以才得以实现多态的。

Java的所有函数,除了被声明为final的,都是用后期绑定。
C++实现多态性,使用关键字virtual,为了引起晚捆绑,使用虚函数。若一个函数在基类被声明为virtual,则所有子类中都是virtual的。对虚函数的重定义成为越位。
Dan1980 2007-03-05
  • 打赏
  • 举报
回复 1
答:通过方法重载和覆盖来体现的。(有的观点认为方法重载不属于多态)
pt870219 2007-03-05
  • 打赏
  • 举报
回复
方法的重载和类的重写

62,615

社区成员

发帖
与我相关
我的任务
社区描述
Java 2 Standard Edition
社区管理员
  • Java SE
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧