zoukankan      html  css  js  c++  java
  • Java之抽象类与抽象方法

    Java之抽象类与抽象方法

    抽象概念

    本篇关键词是抽象,那么何为抽象?百度百科告诉我们,抽象就是概括具体事务共同的方面、本质属性等,而将个别的方面、属性等舍弃的思维过程。在Java里,也就是把各个具体的类中共有的方法提取出来,放到基类之中,而基类并不需要了解子类中该方法具体是怎么实现的,这个基类就是所谓的抽象类,这些不需要知道具体实现方式的方法就是抽象方法

    抽象类体现模板模式的设计,抽象类作为多个子类的通用模板,子类在抽象类的基础上进行扩展、改造,但子类总体上会大致保留抽象类的行为方式。

    抽象类与方法

    让我们结合代码,好好地分析一波:

    package com.my.pac18;
    
    /**
     * @auther Summerday
     */
    /*抽象是很有用的重构工具,能够让共有的方法沿着继承层次向上移动。*/
    public abstract class Shape {
        //抽象类中的初始化块
        {
            System.out.println("Shape.instance initializer");
        }
        //抽象类中的实例变量
        private String color;
        //抽象类中可以有静态方法及属性。
        public static String name="形状";
        public static void output(){
            System.out.println("父类的static方法");
        }
        //计算周长,但是并不知道具体细节的抽象方法
        public abstract double calPerimeter();
        //返回形状的抽象方法
    
        //[修饰符] abstract 返回类型 方法名();
        public abstract String getType();
        //抽象类中的构造器,用于被继承
        public Shape() {
        }
    
        public Shape(String color) {
            System.out.println("Shape.Shape");
            this.color = color;
        }
        //抽象类中的普通实例方法
        public String getColor() {
            return color;
        }
    }
    
    class Circle extends Shape {
        private double radius;
    
        public Circle(String color, double radius) {
            super(color);
            this.radius = radius;
        }
    
        public void setRadius(double radius) {
            this.radius = radius;
        }
    
        @Override
        public double calPerimeter() {
            return 2 * Math.PI * radius;
        }
    
        @Override
        public String getType() {
            return getColor() + "圆形";
        }
    
        //如果从一个抽象类继承,并创建该新类的对象,就必须给基类的所有抽象方法提供定义。
        public static void main(String[] args) {
            Shape.output();
            //Shape是抽象类,不能创建Shape类的对象
            //但是可以让Shape类的引用变量指向其子类的对象
            Shape[] shapes = new Shape[]{new Circle("红色", 5), new Rectangle("绿色", 6, 6)};
            for (Shape p : shapes) {
                
                System.out.println(p.getType() + "的周长为" + p.calPerimeter());
            }
    
        }
    }
    
    class Rectangle extends Shape {
        
        private double length;
        private double width;
    
        public Rectangle() {
            System.out.println("父类的构造器不是为了创建对象,而是供给子类继承的");
        }
    
        public Rectangle(String color, double length, double width) {
            super(color);
            this.length = length;
            this.width = width;
        }
    
        public boolean isSquare() {
            return length == width;
        }
    
        public double calPerimeter() {
            return (length + width) * 2;
        }
    
        public String getType() {
            if (isSquare()) return getColor() + "正方形";
            return getColor() + "长方形";
        }
    }
    
    //测试结果
    父类的static方法
    Shape.instance initializer
    Shape.Shape
    Shape.instance initializer
    Shape.Shape
    红色圆形的周长为31.41592653589793
    绿色正方形的周长为24.0
    

    顺着上面的例子,我们简单地分析一下,抽象类抽象方法到底是啥?又需要注意些啥?

    • 首先定义了一个抽象类:Shape类。可以看到,抽象类由abstract关键字修饰。
    //[修饰符] abstract class 类名
    public abstract class Shape
    
    • 然后再Shape类里定义了两个抽象方法(方法可以被重写,属性不可以,抽象是针对方法而言的):getType()calPerimeter()可以看出抽象方法声明时并没有定义方法体,因为基类中并不需要知道具体的实现方式是怎样,只要知道有这个方法就可以了。你创建一个圆形实例,具体如何算圆形的周长,在圆形类给出抽象方法的定义即可。
    //[修饰符] abstract 返回类型 方法名();
     public abstract double calPerimeter();
     public abstract String getType();
    

    注意事项

    • 如果子类从抽象父类继承而来,那么子类就必须重写父类的抽象方法,给出方法的具体实现形式。你的爸爸说你会算周长,你总得告诉大家周长怎么算吧。但是,如果子类中也重写父类的方法也声明为抽象的话,这时父类方法在子类中的实现也无效了。就好比,你说你也不知道怎么算周长,隔壁小王才是真的会。
    @Override
    public double calPerimeter() {
        return 2 * Math.PI * radius;
    }
    @Override
    public String getType() {
        return getColor() + "圆形";
    }
    
    • 抽象类不能实例化,不能用new来调用抽象类构造器创建抽象类的实例。可以想象,既然是抽象的,创建它的实例就没多大意义,创建具体实现的子类的实例才能解决问题。
    • 抽象类可以用作一种数据类型,可以让抽象类的引用变量指向它的子类对象,很好理解,不这样的话,抽象带来的好处体现在哪呢。通过抽象类的引用变量调用抽象类中的方法,就能够根据动态绑定,在运行时根据引用对象的实际类型执行抽象方法的具体表现形式。
    • 抽象类中是能够有具体实现的成员方法(实例方法和类方法),以及明确的成员属性(实例属性和类属性)。
    //抽象类中的实例方法,子类继承使用。
    public String getColor() {
            return color;
        }
    //抽象类中可以有静态方法和属性,类名.方法(属性)调用。
    public static String name="形状";
    public static void output(){
        System.out.println("父类的static方法");
    }
    
    • 抽象类中还可以有构造器,但是仅仅是为了供给子类继承使用,因为上面讲过,抽象类无法创建实例对象!!!(重点回顾:子类无法继承父类构造器,只是单纯调用父类构造器的初始化代码,继承和调用是不一样的!!!
    //子类
    public Shape() {
        }
    
    public Shape(String color) {
        System.out.println("Shape.Shape");
        this.color = color;
    }
    
    • 抽象类中还有代码块和内部类,关于内部类之后将会学习,暂时就不举例了。
     //抽象类中的初始化块
    {
        System.out.println("Shape.instance initializer");
    }
    
    • 一个类中如果有抽象方法,那么这个类也一定是抽象的。毕竟你的方法定义尚且都不完全,创建你的实例又有啥用呢。但是一个抽象类中可以没有抽象方法,只是单纯不让人创建它的实例对象。

    • 子类是抽象,父类也可以是具体的。如上,Object类是所有类的超类,它是具体的,我们定义的Shape类是可以抽象的。

    • 抽象是实现多态的一种机制,那么不具备多态性的,就不能称之为多态。诸如private、static、final修饰的方法或构造器属于静态绑定,不具备多态性,不能作为抽象的修饰。

    本文是根据资料以及个人测试之后所得,若有错误或者理解不当之处,还望评论区指出,谢谢。

  • 相关阅读:
    MapReduce in MongoDB
    MongoDB的一些基本操作
    谈谈NOSQL
    Java中的反射(1)
    Mybatisの常见面试题
    关于Lombok和自动生成get set方法
    订Pizza(Java)
    美化Div的边框
    爱,死亡和机器人(Love,Death&Robots)
    session与cookie的介绍和两者的区别之其相互的关系
  • 原文地址:https://www.cnblogs.com/summerday152/p/12081810.html
Copyright © 2011-2022 走看看