【Java入门提高篇】Day3 抽象类与接口的比较

简介:   抽象类跟接口都讲完了,现在来做一个比较。  其实说实话,没有多大的可比较性,它们是完全不同的两个东西,它们的抽象不在同一个层级上。但是为了让大家更好的理解,还是做一个比较吧,毕竟它们都很抽象(233)。

  抽象类跟接口都讲完了,现在来做一个比较。

  其实说实话,没有多大的可比较性,它们是完全不同的两个东西,它们的抽象不在同一个层级上。但是为了让大家更好的理解,还是做一个比较吧,毕竟它们都很抽象(233)。

  首先是语法层面上的对比

  1)抽象类跟接口都不能被实例化,因为它们都很虚嘛。但是在访问权限上,两者有一定的区别。

  a、抽象类中的抽象方法(其前有abstract修饰)不能用private、static、synchronized、native访问修饰符修饰。理由很简单,容我慢慢道来。

  抽象方法是没有方法体的,它的目的就是用来继承的,所以如果使用private修饰,不就不能被继承了吗?这就违背了它的设计初衷了,所以不能用private来修饰抽象方法。至于static,用它来修饰的方法可以不实例化就可以直接调用,但是抽象方法没有方法体,使用static修饰就没有意义了。synchronized是用来加锁的,如果修饰类中的方法的话,就相当于用this变量锁,但是抽象类是不能被实例化的,抽象方法也不是在本类中实现而是在子类中实现的,所以锁应该是子类所属,所以抽象方法不能用synchronized关键字修饰;至于native,这个跟abstract关键字本身就是冲突的,abstract声明方法交给子类实现,而native则是交给本地操作系统实现,如果同时出现,那就相当于把实现交给子类,又交给本地操作系统,那最后到底由谁来实现呢?

  综上所述,抽象类中的抽象方法只能用public和protected修饰。

  b.接口中的方法全部为public abstract修饰,不能使用其他修饰符,而且默认情况(不加任何修饰符)下,也是public abstract的,因为接口只能被类实现,不能被类继承,所以不能使用protected修饰,但接口是可以继承接口的。

 

  2)抽象类跟普通类的唯一区别就是不能被实例化,可以有抽象方法,所以它可以有构造函数,静态方法,静态代码块,可以有普通的成员变量和方法。但是接口就不一样了,接口只能声明public abstract的方法和public static final的成员变量。

  3)抽象类本质上还是一个类,只能单继承,一个类只能继承一个抽象类,但可以实现多个接口。

 

  其次是概念上的比较

  1)抽象类跟接口的抽象角度不一样,抽象类一般是对某些具有相似属性和方法的类进行抽象,抽象出一个统一的父类。而接口则更多的是多一组特定行为的抽象,关注的是行为,而具有这些行为的类之间可能并没有太大的关联性。

  比如说,飞机能上天,鸟能上天,你要是厉害一点,应该也能上天(逃),但显然两者之间的关联度不大,如果硬是要给它们插上一个公共的父类的话,似乎不合情理,看起来就像这样:

  

public abstract class Flyer {
    public abstract void fly();
}

  然后定义两个类来继承它:

public class Airplane extends Flyer {

    @Override
    public void fly() {
        System.out.println("Airplane is flying.");
    }
}
public class Bird extends Flyer {

    @Override
    public void fly() {
        System.out.println("Bird is flying.");
    }
}

  好的,现在写一个测试类:

public class Test {
    public static void main(String[] args) {
        Flyer[] flyer = new Flyer[2];
        flyer[0] = new Airplane();
        flyer[1] = new Bird();
        for (Flyer f:flyer){
            f.fly();
        }
    }
}

  运行结果如下:

Airplane is flying.
Bird is flying.

  乍眼一看,好像运行良好,但是仔细想想,将两个关联度很低的类强行插上一个父类,似乎有些不妥,毕竟飞机跟鸟除了都能飞以外,基本没有什么相似的地方了,而且两者的飞行方式,飞行速度和高度都相去甚远,也就是说除了这个fly的方法,其他方法都要在各自的子类实现,而且一个类只能继承一个抽象类,所以Bird类和Airplane类就无法再继承其他类了,这样就反而限制了程序的灵活性。所以,这种时候,还是比较适合使用接口:

public interface IFlyable {
    //声明Fly方法
    void fly();
}

  而此时只需要将Airplane类和Bird类的extends Flyer改成implement Flyable即可。

public class Airplane implements IFlyable {

    //实现Fly方法
    @Override
    public void fly() {
        System.out.println("Airplane is flying.");
    }
}
public class Bird implements IFlyable {
    //实现Fly方法
    @Override
    public void fly() {
        System.out.println("Bird is flying.");
    }
}

  再修改一下Test类:

public class Test {
    public static void main(String[] args) {
        IFlyable[] flyer = new IFlyable[2];
        flyer[0] = new Airplane();
        flyer[1] = new Bird();
        for (IFlyable f:flyer){
            f.fly();
        }
    }
}

  输出如下:

Airplane is flying.
Bird is flying.

  也许从这个栗子还没法明显的看出两者的区别,那么我们再换一个栗子,人可以坐飞机,可以坐火车,还可以坐汽车,只要它们有载人功能即可,那用接口实现如下:

public interface ICarryPassenger {

    //声明载客方法
    void carry(Passenger passenger);
}

  定义一个乘客类,用姓名来区分各个乘客。

public class Passenger {

    private String name;//乘客姓名

    public Passenger(String name){
        this.name = name;
    }

    public String getName() {
        return name;
    }

    //出行方式
    public void travelBy(ICarryPassenger ic){
        ic.carry(this);
    }
}

  分别定义汽车类,火车类,飞机类,它们都实现ICarryPassenger接口,飞机还可以实现IFlyable接口(虽然没有用到。。):

public class Car implements ICarryPassenger {

    int passengerNum;

    //实现carry方法
    @Override
    public void carry(Passenger passenger) {
        System.out.println("Passenger:"+passenger.getName()+" travel by Car.");
        passengerNum++;
        System.out.println("Car carries: "+passengerNum+" passenger.");
    }
}
public class Train implements ICarryPassenger {

    int passengerNum;

    @Override
    public void carry(Passenger passenger) {
        System.out.println("Passenger:"+passenger.getName()+" travel by Train.");
        passengerNum++;
        System.out.println("Train carries: "+passengerNum+" passenger.");
    }
}
public class Airplane implements IFlyable,ICarryPassenger{

    private int passengerNum;//乘客数量

    //实现Fly方法
    @Override
    public void fly() {
        System.out.println("Airplane is flying.");
    }

    //实现carry方法
    @Override
    public void carry(Passenger passenger) {
        System.out.println("Passenger:"+passenger.getName()+" travel by Airplane.");
        passengerNum++;
        System.out.println("Airplane carries: "+passengerNum+" passengers.");
    }
}

  好的,现在我们写一个测试类来进行测试:

public class Test {
    public static void main(String[] args) {
        //有6个乘客想要去旅游,对于旅行方式没有侧重,随机分配交通工具
        Random random = new Random();
        Passenger[] passengers = new Passenger[6];//声明6个乘客
        for (int i=0;i<6;i++){
            passengers[i] = new Passenger("Passenger["+i+"]");
        }
        ICarryPassenger[] icp = new ICarryPassenger[3];//声明3种交通方式
        icp[0] = new Airplane();
        icp[1] = new Car();
        icp[2] = new Train();
        for (int i=0;i<6;i++){
            passengers[i].travelBy(icp[random.nextInt(3)]);
        }
    }
}

  输出如下:

Passenger:Passenger[0] travel by Airplane.
Airplane carries: 1 passengers.
Passenger:Passenger[1] travel by Train.
Train carries: 1 passenger.
Passenger:Passenger[2] travel by Airplane.
Airplane carries: 2 passengers.
Passenger:Passenger[3] travel by Car.
Car carries: 1 passenger.
Passenger:Passenger[4] travel by Train.
Train carries: 2 passenger.
Passenger:Passenger[5] travel by Airplane.
Airplane carries: 3 passengers.

  因为飞机跟火车,汽车之间并没有太大关联,显然无法直接抽象出父类,它们仅有相同的行为,那就是载客,所以使用接口是最合适的。

  至此,本篇讲解完毕,想必通过这一篇的讲解,对于抽象类和接口的区别应该有了更好的理解吧,如果有更好的栗子,欢迎大家留言交流,也欢迎大家继续关注。

 

真正重要的东西,用眼睛是看不见的。
相关文章
|
6天前
|
Java
Java——抽象类和接口
抽象类是一种不能被实例化的类,至少包含一个抽象方法(无实现体的方法),常用于定义一组相关类的共同特征,并强制子类实现特定方法。抽象方法不能被 `static` 或 `final` 修饰,且必须被重写。 接口则是一个完全抽象的类,用于规范类的行为。接口使用 `interface` 关键字定义,不能实例化,并且类与接口之间是实现关系。 内部类是在一个类内定义的类,分为成员内部类、静态内部类、局部内部类和匿名内部类。成员内部类可被修饰符修饰,静态内部类只能访问外部类的静态成员,局部内部类定义在方法内,匿名内部类则隐藏了名字,直接通过 `new` 关键字定义并实现接口或继承类。
12 5
Java——抽象类和接口
|
6天前
|
Java
Java——接口的使用实例
Comparable接口用于自定义类的对象比较。通过实现此接口并重写`compareTo`方法,可以定义自定义类型的比较规则。 接下来介绍了Comparator接口,它提供了一种更灵活的比较方式。通过实现Comparator接口并重写`compare`方法,可以根据不同属性定义不同的比较规则。例如,定义一个`BrandComparator`类来比较汽车的品牌。 最后,介绍了Cloneable接口,用于实现对象的克隆。实现该接口并重写`clone`方法后,可以创建对象的浅拷贝或深拷贝。浅拷贝仅复制对象本身,深拷贝则会递归复制所有成员变量。
13 4
Java——接口的使用实例
|
12天前
|
Java 数据库连接 数据库
Java服务提供接口(SPI)的设计与应用剖析
Java SPI提供了一种优雅的服务扩展和动态加载机制,使得Java应用程序可以轻松地扩展功能和替换组件。通过合理的设计与应用,SPI可以大大增强Java应用的灵活性和可扩展性。
44 18
|
9天前
|
Java 开发者
Java的接口详解
Java接口是编程中的一种重要特性,用于定义方法签名而不提供具体实现,作为类之间的契约,使不同类能以统一方式交互。接口使用`interface`关键字定义,可包含方法声明和常量。类通过`implements`关键字实现接口,并可同时实现多个接口,解决多重继承问题。接口中的方法默认为抽象方法,变量默认为`public static final`。Java 8引入了默认方法和静态方法,增强接口功能。接口广泛应用于回调机制和多态性实现,有助于构建更灵活和可维护的代码结构。
|
19天前
|
Java
盘点java8 stream中隐藏的函数式接口
`shigen`是一位坚持更新文章的博客作者,记录成长历程,分享认知见解,留住感动瞬间。本文介绍了函数式接口的概念及其在Java中的应用,包括`Comparator`、`Runnable`、`Callable`等常见接口,并详细讲解了`Function`、`Predicate`、`Consumer`、`Supplier`和`Comparator`等函数式接口的使用方法及应用场景,展示了如何利用这些接口简化代码并提高编程效率。**个人IP:shigen**,与shigen一起,每天进步一点点!
29 0
盘点java8 stream中隐藏的函数式接口
|
9天前
|
Java 程序员
Java中的异常处理:从入门到精通
在Java编程的世界中,异常处理是保持程序稳定性和可靠性的关键。本文将通过一个独特的视角—把异常处理比作一场“捉迷藏”游戏—来探讨如何在Java中有效管理异常。我们将一起学习如何识别、捕捉以及处理可能出现的异常,确保你的程序即使在面对不可预见的错误时也能优雅地运行。准备好了吗?让我们开始这场寻找并解决Java异常的冒险吧!
|
20天前
|
Java 程序员 UED
Java 中的异常处理:从入门到精通
【8月更文挑战第31天】在Java编程的世界中,异常处理是保持应用稳定性的重要机制。本文将引导你理解异常的本质,学会如何使用try-catch语句来捕获和处理异常,并探索自定义异常类的魅力。我们将一起深入异常的世界,让你的代码更加健壮和用户友好。
|
20天前
|
Java 数据库连接 开发者
Java中的异常处理:从入门到精通
【8月更文挑战第31天】 在编程世界中,错误和异常就像是不请自来的客人,总是在不经意间打扰我们的程序运行。Java语言通过其异常处理机制,为开发者提供了一套优雅的“待客之道”。本文将带你走进Java异常处理的世界,从基础语法到高级技巧,再到最佳实践,让你的程序在面对意外时,也能从容不迫,优雅应对。
|
28天前
|
Java 开发者
Java抽象类与接口的正确使用姿势!别再写错代码了!
Java抽象类与接口的正确使用姿势!别再写错代码了!
34 3
|
27天前
|
Java 程序员
详解Java中的抽象类与接口的区别
【8月更文挑战第24天】
22 0