为了账号安全,请及时绑定邮箱和手机立即绑定

Java面向对象之多态性

标签:
Java
一、多态性

多态是指一个对象可以拥有多种不同的形态,继承是实现多态的基础。

1.1 引用多态和方法多态
引用多态:父类引用可以指向本类的对象,也可以指向子类的对象
方法多态:
1、创建本类对象时,调用的方法为本类方法;
2、创建子类对象时,调用的方法为子类重写或继承的方法。

首先建立父类Animal,包含一个eat()方法,如下代码所示:

public class Animal {
    public void eat(){
        System.out.println("动物可以吃东西");
    }
}

然后建立子类Dog继承父类Animal,并重写eat()方法。

public class Dog extends Animal{
    public void eat(){
        System.out.println("狗狗可以吃东西");
    }
}

最后建立一个Cat类继承Animal,并独有getMouse()方法。

public class Cat extends Animal{
    public void getMouse(){
        System.out.println("猫会抓老鼠");
    }
}

通过测试类进行测试发现,父类的引用不能调用子类独有的方法。

public class InitTest {
    public static void main(String[] args) {
        Animal obj1 = new Animal();//父类的引用指向本类对象
        Animal obj2 = new Dog();//父类的引用指向子类对象
        Animal obj3 = new Cat();
        obj1.eat();//动物可以吃东西
        obj2.eat();//20岁的狗狗可以吃东西
        obj3.eat();//动物可以吃东西
    }
}
1.2 引用类型转换
向上类型转换(隐式或自动类型转换):子类转换为父类对象
向下类型转换(强制类型转换):父类转换为子类,有风险

向上类型转换好比是把杯子中的水倒进壶里,不会存在风险;而向下类型转换则是把壶里的水倒进杯子里,可能会溢出。为了避免这种风险的发生,可以使用instanceof关键字进行判断。如下代码所示:

public class InitTest {
    public static void main(String[] args) {
        Animal obj = new Dog();//向上类型转换
        if(obj instanceof Dog){
            Dog dog = (Dog)obj; //向下类型转换
        }else{
            System.out.println("不能进行Dog类型转换");
        }
        if(obj instanceof Cat){
            Cat cat = (Cat)obj;
        }else{
            System.out.println("不能进行Cat类型转换");
        }
    }
}

程序能够正常运行,且不能进行Cat类型转换。

二、抽象类与接口
2.1 抽象类

当某个父类只知道子类应该包含这些方法却不知如何实现时,需要用到抽象类。抽象类用abstract关键字修饰,它限制子类必须有哪些方法,但不关注实现。抽象类中可以包含普通方法,也可以没有方法。

例:利用抽象类编写程序,求矩形和圆的周长和面积。

实现步骤:
1、抽象出矩形和圆形的抽象类Shape作为父类,并定义要实现的方法求周长premeter()和求面积area();
2、分别创建子类Rectangle和Circle,继承抽象父类Shape;
3、根据相关图形的求解算法,分别实现父类的抽象方法。
public abstract class Shape {
    public abstract double perimeter();
    public abstract double area();
}
public class Rectangle extends Shape {
    double length;
    double width;

    public Rectangle(double length,double width){
        this.length = length;
        this.width = width;
    }

    @Override
    public double perimeter() {
        return 2*(length+width);
    }

    @Override
    public double area() {
        return length*width;
    }

}
public class Circle extends Shape {
    double radius;
    final double PI = 3.14;

    public Circle(double radius){
        this.radius = radius;
    }

    @Override
    public double perimeter() {
        return 2*PI*radius;
    }

    @Override
    public double area() {
        return PI*radius*radius;
    }

}

这里,由于不同的形状求解周长和面积时,需要的参数个数不同,在定义抽象方法中无法统一给出参数,可以在子类中定义变量,通过初始化构造方法进行参数的传递。下面是测试方法代码:

public class ShapeTest {
    public static void main(String[] args) {
        Shape s1 = new Rectangle(2,4);
        Shape s2 = new Circle(3.9);
        System.out.println("矩形的周长:"+s1.perimeter());
        System.out.println("矩形的面积:"+s1.area());
        System.out.println("圆形的周长:"+s2.perimeter());
        System.out.println("圆形的面积:"+s2.area());
    }
}
2.2 接口

接口是一种特殊的类,由全局变量和公共方法组成。它定义了某一批类的规范,而不关心这些类的内部数据和实现细节,只规定这些类里必须提供某些方法。

接口里的属性是常量,默认会加上public static final三个关键字,接口里的方法只能是抽象方法,默认会加上public abstract关键字。

一个类可以实现多个接口,但类必须是先继承,然后实现接口。

例:傻瓜机和智能机继承了父类手机,父类的抽象方法里面都定义了打电话和发短信的功能,而智能机和PSP都有玩游戏的功能。此时PSP不具备手机的特征所以不宜继承手机类,而可以将玩游戏定义为一个接口,由智能机和PSP实现。

public interface IPlayGame {
    public void playGame();
}
public class SmartPhone extends Telphone implements IPlayGame{

    @Override
    public void call() {
        System.out.println("SmartPhone可以打电话");
    }

    @Override
    public void message() {
        System.out.println("SmartPhone可以发短信");
    }

    @Override
    public void playGame() {
        System.out.println("玩游戏的功能");
    }

}
public class PSP implements IPlayGame{
    @Override
    public void playGame() {
        System.out.println("玩游戏的功能");
    }
}
public class InterfaceTest {

    public static void main(String[] args) {
        IPlayGame ip1 = new SmartPhone();
        IPlayGame ip2 = new PSP();
        ip1.playGame();
        ip2.playGame();
    }

}

接口还可以通过匿名内部类的方式实现,即没有名字的内部类。此方式常用于高级开发中。

    public static void main(String[] args) {
        IPlayGame ip3 = new IPlayGame(){
        @Override
        public void playGame() {
            System.out.println("使用匿名内部类的方式实现接口");
        }
    };
    ip3.playGame();
}

注:文章根据本站相关课程的学习笔记整理而来,有错误或其他任何问题敬请指正,欢迎共同学习交流!

点击查看更多内容
TA 点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
软件工程师
手记
粉丝
36
获赞与收藏
349

关注作者,订阅最新文章

阅读免费教程

  • 推荐
  • 1
  • 收藏
  • 共同学习,写下你的评论
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消