java多态

多态在Java中表现为不同的对象执行同一行为时产生不同的状态。实现多态需要继承、方法重写、通过父类引用调用以及向上转型。动态绑定使得父类引用调用子类重写的方法成为可能,提高了代码灵活性,降低了圈复杂度。然而,多态可能导致运行效率下降,且属性和构造方法不具备多态性。向上转型时,父类引用无法访问子类特有的成员。
摘要由CSDN通过智能技术生成
多态的概念:通俗来说,就是多种形态, 具体点就是去完成某个行为,当不同的对象去完成时会产生出不同 的状 态。
java 中要实现多态,必须要满足如下几个条件,缺一不可:
1. 必须在继承体系下
2. 子类必须要对父类中方法进行重写
3. 通过父类的引用调用重写的方法
4.向上转型
class Animal{
   public String name;
   public int age;
    public  void eat(){
        System.out.println(name+"在吃饭");
    }
}
class Dog extends Animal{
    public  void wang(){
        System.out.println(name+"在汪汪叫");
    }
}
class Bird extends Animal{
    public String wing;
    public  void fly(){
        System.out.println(name+"在飞");
    }
}
public class test {
    public static void main(String[] args) {
        Dog dog=new Dog();
        dog.name="旺财";
        dog.wang();
    }
}

 向上转型:因为所有的子类都是继承父类的,

把子类对象赋给父类对象叫向上转型。

理论上说等号两边变量类型一定要相同,而这两个类型不一样,因为它俩之间构成父子类关系,

 

 下面这样向上转型也是可以的。

 看个例子。

 我们发现下面这样调用子类方法会报错。这是为什么?

因为animal1中没有wang(),通过引用调用时,只能调用animal1类包含的方法,类中没有的方法是调用不了的。

当发生向上转型后,父类的引用只能访问父类自己的成员,不能访问子类特有的成员。

 下面这种函数传参也属于向上转型。

下面这种返回值也属于向上转型。 

重写:

1.方法名称相同

2.参数列表相同

3.返回值相同

@Override可以检查是否进行了重写。

 @Override叫注解,这个注解意思是当前方法是重写的。

class Animal{
   public String name;
   public int age;
    public  void eat(){
        System.out.println(name+"在吃饭");
    }
}
class Dog extends Animal{
    public  void wang(){
        System.out.println(name+"在汪汪叫");
    }
    @Override
    public void eat(){
        System.out.println(name+"正常吃狗粮");
    }
}
class Bird extends Animal{
    public String wing;
    public  void fly(){
        System.out.println(name+"在飞");
    }
    @Override
    public void eat(){
        System.out.println(name+"正常吃鸟粮");
    }
}
public class test {
    public static void func(Animal animal) {

    }
    public static Animal func2(Animal animal)
    {
     return new Dog();
    }
    public static void main(String[] args) {
        Animal animal1=new Dog();
        animal1.name="旺柴";
        animal1.eat();
//        animal1.wang();
        Animal animal2=new Bird();
        animal2.name="财旺";
        animal2.eat();
        func(animal1);
    }
  
}

 我们发现此时调用了子类的eat(),之前说父类只能调用父类的方法,这里为什么调用了子类方法?

这个叫作动态绑定。

编译时还是调用animal中的eat(),运行时绑定到了子类的eat().(其实是把要调用的方法地址改变了)

动态绑定:

1.向上转型

2.重写

3.父类引用调用父类和子类重写的方法

静态绑定:编译时知道调用哪个方法了。比如重载。

super是个关键字,不代表父类引用,就是在书写上能一目了然,知道这是父类的。 

下面再看个例子。

对于此时的animal来说,它不知道自己将来引用哪个对象,但是我们能确定animal将来指向的对象不同,执行的方法就不一样,这个也就是多态。

父类引用引用的对象不一样的时候,表现出的行为是不一样的,这也就是多态。

 注意:

1.private修饰的方法不能重写。

2.static修饰的方法不能被重写。

3.子类的访问修饰限定权限要大于大于父类的权限。

4.被final修饰的方法不能被重写。此时这个方法叫密封方法。

重写原则 

对于已经投入使用的类,尽量不要进行修改。最好的方式是:重新定义一个新的类,来重复利用其中共性的内容, 并且添加或者改动新的内容。

 例如:若干年前的手机,只能打电话,发短信,来电显示只能显示号码,而今天的手机在来电显示的时候,不仅仅 可以显示号码,还可以显示头像,地区等。在这个过程当中,我们不应该在原来老的类上进行修改,因为原来的类,可能还在有用户使用,正确做法是:新建一个新手机的类,对来电显示这个方法重写就好了,这样就达到了我们当今的需求了

向上转型优点:让代码实现更灵活。

向上转型缺点:不能调用子类特有的方法。

  • 向下转型: 

把父类对象给子类。

 向下转型非常不安全。

下面就错了

因为实际给bird的是Dog的对象,里面没有fly()。 

animal1范围大于bird这样会出错。

 为了解决这个问题我们可以判断一下。

这样就不报错了。这里是判断animal1是否引用了Bird对象。

 我们看个例子。

class Shape{
    public void draw(){
        System.out.println("画图形");
    }
}
class Rect extends Shape{
    @Override
    public void draw() {
        System.out.println("画矩形");
    }
}
class Cycle extends Shape{
    @Override
    public void draw() {
        System.out.println("画圆");

    }
}
public class draw {
    public static void drawmap(Shape shape)
    {
        shape.draw();
    }

    public static void main(String[] args) {
        Rect rect=new Rect();
        Cycle cycle=new Cycle();
        drawmap(rect);
        drawmap(cycle);
    }
}

 使用多态的好处:

1. 能够降低代码的 " 圈复杂度 ", 避免使用大量的 if - else
什么叫 " 圈复杂度 " ?
圈复杂度是一种描述一段代码复杂程度的方式 . 一段代码如果平铺直叙 , 那么就比较简单容易理解 . 而如
果有很多的条件分支或者循环语句 , 就认为理解起来更复杂 .
因此我们可以简单粗暴的计算一段代码中条件语句和循环语句出现的个数 , 这个个数就称为 " 圈复杂度 ".
如果一个方法的圈复杂度太高 , 就需要考虑重构 .
不同公司对于代码的圈复杂度的规范不一样 . 一般不会超过 10 .
例如我们现在需要打印的不是一个形状了 , 而是多个形状 . 如果不基于多态 , 实现代码如下 :
如果使用使用多态 , 则不必写这么多的 if - else 分支语句 , 代码更简单 .
2. 可扩展能力更强
如果要新增一种新的形状 , 使用多态的方式代码改动成本也比较低 .
对于类的调用者来说 (drawShapes 方法 ), 只要创建一个新类的实例就可以了 , 改动成本很低 .
而对于不用多态的情况 , 就要把 drawShapes 中的 if - else 进行一定的修改 , 改动成本更高.
多态缺陷:代码的运行效率降低
1. 属性没有多态性
当父类和子类都有同名属性的时候,通过父类引用,只能引用父类自己的成员属性
2. 构造方法没有多态性

下面例子中,为什么Shape数组可以存不同类型的值?

 这个过程发生了向上转型。Shape里面的每一个类型就是Shape类型,

那么就相当于这样Shape shape=new Shape().

父类类型的数组能存放子类类型的对象。

 父类构造方法中如果调用子类和父类重写的方法,那么会调用子类的方法。

 而且此时num等于0,因为正在调用父类构造函数,父类都没有初始化完成,因为继承中是父类先构造,所以此时num是默认值0。

所以以后不要这么写。 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

南种北李

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值