Java-基础篇-抽象类和接口

1.抽象类

1.1 抽象类概念

抽象方法是只声明而未实现的方法,所有的抽象方法必须使用abstract关键字声明,包含抽象方法的类也必须使用abstract class声明。

1.1.1 抽象类定义规则

  1. 抽象类和抽象方法都必须用abstract关键字来修饰;
  2. 抽象类不能直接实例化,即不能使用new关键字去产生对象;
  3. 抽象类定义时抽象方法只需声明,而不需实现;
  4. 含有抽象方法的类必须被声明为抽象类,抽象类的子类必须覆写所有的抽象方法后才能被实例化,否则这个子类还是个抽象类。

1.2 抽象类声明

访问权限  abstract class 类名 {
    // 声明成员变量
    访问权限 变量类型 成员变量名;
    // 声明一般方法
    访问权限 返回类型 方法名(){
        // ...
    }
    // 声明抽象方法
    abstract 返回类型 方法名();
}

实例

/**
 * description:
 *
 * @author : Mr.Liuqh
 * @since : 2019-10-31 18:51
 */
public abstract class People {
    // 声明成员变量
    private String name;

    // 声明一般方法
    public String getName(){
        return this.name;
    }

    // 声明抽象方法
    abstract String like();
}
由上可知:抽象类的定义就是比普通类多了一些抽象方法的定义而已。虽然定义了抽象类,但是抽象类却不能直接使用。
People girl = new People(); // 这种调用会报错,People 是抽象的,无法实例化

1.2.1 抽象类的使用原则

如果说一个类的对象可以被实例化,那么就表示这个对象可以调用类中的属性或者是方法,但是抽象类中存在抽象方法,而抽象方法没有方法体,没有方法体的方法无法使用。

对于抽象类的使用原则如下:

  • 抽象类必须有子类,子类使用extends继承抽象类,一个子类只能够继承一个抽象类;
  • 子类(如果不是抽象类)则必须覆写抽象类之中的全部抽象方法;
  • 若想实例化抽象类的对象,则可以使用子类进行对象的向上转型来完成。

1.3 抽象类特征

1.3.1 抽象类中可以有构造方法

与一般类相同,在抽象类中也可以拥有构造方法,但是这些构造方法必须在子类中被调用,并且子类实例化对象的时候依然满足类继承的关系,先默认调用父类的构造方法,而后再调用子类的构造方法,毕竟抽象类之中还是存在属性的,只不过这个抽象方法无法直接被外部实例化对象的时候所使用。

代码示例:

//定义一抽象类Person
abstract class Person
{
    String name;
    int age;
    String occupation;
    public Person(String name,int age,String occupation) //定义构造函数
    {
        this.name=name;
        this.age=age;
        this.occupation=occupation;
    }
    public abstract String talk();
    //声明一个抽象方法
}

//声明抽象类的子类
class Student extends Person    
{
    public Student(String name,int age,String occupation){
        //在这里必须明确调用抽象类中的构造方法
        super(name,age,occupation);
    }

    //覆写talk()方法
    public String talk() {
        return "学生——>姓名:"+this.name+",年龄:"+this.age+",职业:"+this.occupation+"!";
    }
}

// 测试
class AbstractConstructor
{
    public static void main(String[] args){
        //创建对象s
        Student s=new Student("张三",18,"学生");

        //调用被覆写过的方法
        System.out.println(s.talk()); // 输出:学生——>姓名:张三,年龄:18,职业:学生! 

    }
}

从程序中可以看到,抽象类也可以像普通类一样,有构造方法、一般方法和属性,更重要的是还可以有一些抽象方法,需要子类去实现,而且在抽象类中声明构造方法后,在子类中必须明确调用。

1.3.2 抽象类不能够使用final定义。

使用final定义的类不能有子类,而抽象类使用的时候必须有子类,这是一个矛盾的问题,所以抽象类上不能出现final定义。

1.3.3 在外部抽象类上无法使用static声明

在外部抽象类上无法使用static声明,但是内部抽象类却可以使用static定义,使用static定义的内部抽象类就表示一个外部类。

代码

abstract class Book{
    //抽象方法
    public abstract void print();
    //静态内部抽象类
    static abstract class CD{
      //抽象方法
      public abstract void get();         

    }
}
//继承抽象类
class JavaCD extends Book.CD {
    public void get(){
        System.out.println("java学习");
    }
}

public class StaticInnerAbstractClass{
    public static void main(String[]  args){
        //实例化对象
        Book.CD cd=new JavaCD();
        cd.get(); // 输出:java学习
    }
}

2. 接口

接口(interface)是Java所提供的另一种重要技术,是一种特殊的类,它的结构和抽象类非常相似,也具有数据成员与抽象方法,但它与抽象类又有不同,并且Java 8中又添加了新特性。

2.1 接口的基本概念

  • 接口里的数据成员必须初始化,且数据成员均为常量,常见的是全局变量。
  • 接口里的方法为abstract,接口不能像抽象类一样定义一般的方法,需定义“抽象方法”。
Java8中为避免在接口中添加新方法后要修改所有实现类,允许定义默认方法,即default方法,也可以称为Defender方法,或者虚拟扩展方法(Virtual extension methods)。
Default方法是指,在接口内部包含了一些默认的方法实现(也就是接口中可以包含方法体,这打破了Java之前版本对接口的语法限制),从而使得接口在进行扩展的时候,不会破坏与接口相关的实现类代码。

2.2 接口定义

声明格式:

public interface 接口名称 {
    //数据成员必须赋初值
    final 数据类型 成员名称=常量;
    //抽象方法,注意没有定义方法主体
    abstract 返回数据类型 方法名称(参数…);

    //默认方法,包含方法体
    default 返回值的数据类型方法名称(参数…){
        // 方法体...
    }
}

示例:

  • 无默认方法的接口
interface MyInterface{
    // 数据成员必须赋初值
    static final String NAME="Mr.Liu" 
    // 抽象方法
    abstract String getName();
}
  • 有默认方法的接口
public interface MyInterface{
    // 数据成员必须赋初值
    static final String NAME="Mr.Liu" 
    // 抽象方法
    abstract String getName();
    // 默认方法
    default  String say(){
        System.out.println("Hello!")
    }
}

2.3 接口使用原则

  • 接口必须有子类,子类依靠implements关键字可以同时实现多个接口;
  • 接口的子类(如果不是抽象类)则必须覆写接口之中的全部抽象方法;

  • 接口可以利用对象多态性,利用子类实现对象的实例化。

  • 接口与一般类一样,本身也具有数据成员与方法,但数据成员一定要赋初值,且此值不能再更改
  • 接口中除default方法外必须都是抽象方法,所以接口定义格式中,抽象方法声明的关键字abstract是可以省略的。

  • 一个类实现多个接口时,若接口有默认方法,不能出现同名的默认方法

  • 接口可以继承多个接口

3.抽象类和接口的区别


3.1 共同点

  • 都是抽象类型

  • 都可以有实现方法(以前接口不行)

  • 都可以不需要实现类或者继承者去实现所有方法。(以前不行,现在接口中默认方法不需要实现者实现)

3.2 不同点

  • 抽象类不可以多重继承,接口可以(无论是多重类型继承还是多重行为继承)

  • 抽象类和接口所反映出的设计理念不同。其实抽象类表示的是”is-a”关系,接口表示的是”like-a”关系

  • 接口中定义的变量默认是public static final 型,且必须给其初值,所以实现类中不能重新定义,也不能改变其值;抽象类中的变量默认是 friendly 型,其值可以在子类中重新定义,也可以重新赋值

    如果一个类、类属变量及方法不以这三种修饰符来修饰,它就是friendly类型的,那么包内的任何类都可以访问它,而包外的任何类都不能访问它(包括包外继承了此类的子类)

总体来说,抽象类和接口在很大程度上都是可以互相替换使用的,但就是由于抽象类本身具备单继承局限,所以当抽象类和接口全部都可以使用的时候优先考虑接口,因为接口没有单继承局限,并且在Java8中接口可以设定默认方法,在一定程度上避免代码重复,利于后期的维护。


转载请注明来源,欢迎指出任何有错误或不够清晰的表达。

文章标题:Java-基础篇-抽象类和接口

文章字数:2.2k

本文作者:猿码记

发布时间:2019-11-05 19:11

原始链接:liuqh.icu/2019/11/05/java-9-抽象类和接口/

版权声明: 转载请保留原文链接及作者。

目录
×

看您心情~~