Java——多态

1.多态简介

        多态,字面意思来看就是多种形态,即当去执行某个行为,会因为对象的不同而产生不同的效果。在java语言当中就是根据对象的性质不同,对同一个方法进行调用时,得到了不同的执行结果。

        在理解多态中必须时刻牢记三个条件:

(1)存在继承关系,并且发生向上转型。

(2)子类对父类的方法进行重写。

(3)通过父类对象的引用去调用这个重写的方法。

        在满足了如上三个条件后,就会发生动态绑定,而动态绑定则是多态的基础。

         接下来,我们将会对刚才涉及到的概念进行一一解释。

2.向上转型和向下转型

        我们知道,在继承关系中存在父类和子类,父类对象和子类对象之间可以存在类型转换。当我们将子类对象赋值给父类对象,这就构成了向上转型;当我们将父类对象赋值给子类对象,这就构成了向下转型。

        需要指出的是,当对象进行向上转型后,就不可以再调用子类特有的方法了。如果想要调用子类方法,那么就需要进行向下转型还原为子类对象。

2.1 向上转型

        向上转型:将子类对象赋值给父类对象

        向上转型常见发生情况:

直接赋值

class Base{}

class Derived extends Base{}

public class Test {
    public static void main(String[] args) {
        Base b1 = new Derived();    //直接赋值
    }
}

作为方法的参数,将子类对象实参传给父类对象形参

class Base{}

class Derived extends Base{}

public class Test {
    public static void func(Base b){}

    public static void main(String[] args) {
        Derived d = new Derived();
        func(d);    //传参
    }
}

作为返回值,将子类对象以父类对象的形式返回

class Base{}

class Derived extends Base{}

public class Test {

    public static Base func(){
        Derived d = new Derived();
        return d;
    }
    public static void main(String[] args) {
        func(); //返回值
    }
}

2.2 向下转型

        向下转型:将父类对象赋值给子类对象

        向下转型具体来说就是将一个子类对象向上转型后的父类对象还原为其原来的子类对象。由于向下转型存在着安全性问题,所以向下转型的时候需要强制类型转换。

class Base{}

class Derived extends Base{}

public class Test {
    public static void main(String[] args) {
        Derived d = (Derived) new Base();
    }
}

        不完全主要是因为父类可以被很多类继承,所以父类拥有众多子类,无法肯定将要向下转型的对象是不是就是指定的子类,所以存在安全性问题。这时候就需要使用instanceof关键字对对象进行检查,如果是指定子类向上转型的对象,则表达式为真。

class Base{}
class Derived1 extends Base{
    public void func(){
        System.out.println("Derived1");
    }
}
class Derived2 extends Base{
    public void func(){
        System.out.println("Derived2");
    }
}
public class Test {
    public static void main(String[] args) {
        Base b1 = new Derived1();
        if(b1 instanceof Derived2){
            Derived2 d = (Derived2)b1;
            d.func();
        }
        if(b1 instanceof Derived1){
            Derived1 d = (Derived1)b1;
            d.func();
        }
    }
}

Derived1

3. 重写

3.1 重写的实现

        重写就是子类对父类的方法进行重新编写,所以要求重写的方法除了方法体以外,函数名、返回值、参数等都不可以改变。

class Base{
    public void func(){
        System.out.println("Base::func()");
    }
}

class Derived extends Base{
    @Override
    public void func(){
        System.out.println("Derived::func()");
    }
}

        在实现重写的过程中存在如下需要注意的关键点:

①重写是父类和子类方法之间的关系,所以重写必然依附于继承关系

②子类中重写的方法和父类的方法相比,返回值、方法名、参数列表必须完全一致

③被重写的方法的访问修饰限定符,子类必须大于等于父类的

private修饰的方法不可以重写,因为在子类中都无法访问到这个方法。

static修饰的方法不可以重写

final修饰的方法不可以重写,这也正是final修饰方法的作用。

构造方法不可以被重写

⑧代码中存在一行:@Override 。这是一个注解,他是用于提示我们也是在告诉编译器接下来写的方法是一个重写方法,所以编译器就会进行检查,当发现不是重写方法的时候就会报错。

3.2 重写的使用

        那么重写有什么作用呢?我们可以通过代码进行测试:

class Base{
    public void func(){
        System.out.println("Base::func()");
    }
}

class Derived extends Base{
    public void func(){
        System.out.println("Derived::func()");
    }
}

public class Test {
    public static void main(String[] args) {
        Base b = new Derived();
        b.func();
    }
}

Derived::func()

        可以发现当我们使用向上转型后的对象调用重写方法的时候,对象调用了子类的方法。这个向上转型对象调用子类重写方法的过程,由于在编译时不确定方法的行为,所以需要等到程序运行时才能确定,因此称为动态绑定。同样的,一般的函数(包括函数重载),都是可以在编译时根据所传递的信息(如函数名、参数)等,就可以确定所调用的方法,因而称谓静态绑定。

        在有了如上基础后,我们就可以再来理解一下之前的toString。首先需要明确的一点,所有类的父类都默认是Object(一个java中提供的类),而Object中有toString方法。当我们的类(子类)不实现toString的时候,就会默认调用父类的toString方法;当我们自己写了toString方法后就构成了重写,所以这个时候再调用toString,根据继承关系子类优先原则,就会先调用子类的方法。所以会发现,toString方法的访问限定符只可以是public,这就是重写在发挥作用。

4.总结

        在父类的构造方法中是可以调用子类的重写方法的,但是由于子类初始化在父类之后,所以会产生问题,不建议这样使用。

        通过以上的介绍,可以为多态下一个定义了,多态就是对不同对象尽管看似调用了同一种方法,但是由于对象是向上转型的,方法是存在重写的,所以相当于父类对象调用了不同的子类的重写的方法,因此形成了多态。

  • 19
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

犀利卓

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

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

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

打赏作者

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

抵扣说明:

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

余额充值