例子
class Person{} class Boy extends Person{} class Girl extends Person{}
生活上:
程序上:
我们一般在创建对象的时候
Boy boy = new Boy(); Girl girl = new Girl();
而多态的创建
//父类的引用指向子类的对象 //前面是父类声明 = 新建一个子类的对象 Person p = new Boy();
我们可以直接输出这个父类看一下
Person p = new Boy(); System.out.println(p);
运行结果:
Boy@2503dbd3
发现我们运行的时候识别到了它是Boy类
也就是说我们在编译前它被识别为父类,而编译后也就是运行的时候就被识别为子类了
这就是多态
在编译的时候创建父类引用指向子类的对象
而特性是运行的时候就已经是子类的对象了
我们来看这样一个场景
我们人都是要吃饭的,都有吃饭这个行为
class Person { public void eat(){ System.out.println("是人就得吃饭"); } }
虽然人都得吃饭,但是针对到男孩,女孩在吃饭上是有一定的差异的。
就需要重写一下父类的方法了
//男孩 class Boy extends Person{ public void eat(){ super.eat(); System.out.println("男孩要力气,得吃肉"); } } //女孩 class Girl extends Person{ public void eat() { super.eat(); System.out.println("女孩要苗条,得少吃"); } }
我们假设一个食堂类,食堂里是男孩,女孩吃饭的地方
class Canteen{ public void meal(Boy boy){ System.out.println("到食堂干饭"); boy.eat(); } public void meal(Girl girl){ System.out.println("到食堂干饭"); girl.eat(); } }
那么我们想要男孩,女孩都能进去吃饭,我们就需要进行重载,才能识别所有的参数
测试代码:
Canteen c = new Canteen(); Boy boy = new Boy(); c.meal(boy);
但是这样就会有一个问题
我们这里是只有男孩和女孩这两个类,重载了一次,代码开始有些重复了。
我们之前说过代码重复,我们就需要找到合适的方法解决重复的代码
为什么使用多态
我们一开始说过,父类引用指向子类的对象是多态,该如何运用呢
Person p = new boy(); p.eat();
我们发现调用的是子类重写父类的方法
实际上多态的使用就这么简单
系统会自动进行类型转换
通过父类引用变量调用的方法是子类覆盖或继承的子类方法,不是父类的方法
通过父类引用变量无法调用子类特有的方法
将我们参数从子类改为父类
代码如下
class Canteen{ public void meal(Person person){ System.out.println("到食堂干饭"); boy.eat(); } }
我们测试传参,测试代码不需要更改
Canteen c = new Canteen(); Boy boy = new Boy(); c.meal(boy); //测试的参数不需要改变
也是能够实现的,但是不需要重载
这个能够实现是因为形参Person person就是创建了一个父类的引用
当我们传参的时候,就是使父类指向了子类的对象
这个就更简单了
我们先写一个方法
代码如下
public Person getPerson(String choose){ if(choose.equals("boy")){ return new Boy(); }else if(choose.equals("girl")){ return new Girl(); }else{ System.out.println("输入有误"); return null; } }
当我们调用该方法的时候,通过参数选择,返回男孩或者女孩。
返回的都是一个Person类型的,但是却是一个父类的引用指向子类的对象
刚才说过,父类引用指向子类的对象,是向上转型。
//左边是父类 右边是子类 //和以前学习基础类型时候的默认强转很像 //int a = short Person p = new Boy();
有向上转型就有想下转向
//向下转型的注意点是:能被转为子类的父类,一定是父类引用指向子类的对象 //能转成功,也必须是父类指向的就是这个子类才行 //同样和基础类型的强转很像 //short a = (short)int Boy b =(Boy)p;
instanceof
关键字我们是已知当前这个对象是父类指向的某个子类
所以直接能够将父类向下转型为子类
但是我们可以看下面代码
Person p = new Boy(); Girl g = (girl)p;
运行之后就告诉我们,是没办法将Boy类转为Girl类
我们明显的发现一定需要转为对应的子类才能成功。
所以我们需要一些判断手段
boolean a = p instanceof Boy; //我们发现返回的是true
由此我们可以看出来
[对象名] instanceof [类]; //判断对象是否是该类的 //是返回true 否返回false
这样,我们就可以直接先判断再强转了
if(p instanceof Boy){ Boy b =(Boy)p; }
可保万无一失
我们本来今天就是为了使用父类引用子类的对象去操作,为什么还需要进行强转回去呢
我们发现,使用多态没有办法调用子类独有的方法,当我们需要调用子类独有的方法的时候,我们进行强转然后再调用
面向对象的三大特性:封装、继承、多态
封装是隐藏对象的属性和实现细节
继承是软件可重用性的一种表现
多态是具有表现多种形态的能力的特征