面试题:接口和抽象类的区别
需求:接口是否可继承接口? 抽象类是否可实现(implements)接口? 抽象类是否可继承实体类(concrete class)? 抽象类中是否可以有静态的main方法?
先说明二者的定义,然后聊聊需求,最后分析二者的区别。
含有abstract修饰符的类即为抽象类,抽象类不能创建实例对象。含有抽象方法的类必须定义为abstract class。在abstract class中,方法不必是抽象的,但是抽象方法必须在具体子类中实现,所以,不能有抽象构造方法或抽象静态方法。子类如果没有实现抽象父类中的所有抽象方法,则必须定义为abstract类型。抽象类需要被继承才能使用,而被final修饰的类无法被继承,所以abstract和final是不能共存的。
接口(interface)可以说成是抽象类的一种特例,接口中的所有方法都必须是抽象的。接口中的方法定义默认为public abstract类型,接口中的成员变量类型默认为public static final。
接口可以继承接口。
抽象类可以实现(implements)接口。
抽象类可以继承实体类。但和实体类的继承一样,也要求父类可继承,并且拥有子类可以访问到的构造函数。其实Object就是个实体类,Java的API文档里,每个抽象类的条目里都明确写着直接或间接继承自Object,所以这点是没有疑问的。
抽象类中可以有静态的main方法。下面分析二者的区别。
备注:只要明白了接口和抽象类的本质和作用,这些问题都很好回答,你想想,如果你是java语言的设计者,你是否会提供这样的支持,如果不提供的话,有什么理由吗?如果你没有道理不提供,那答案就是肯定的了。
只有记住抽象类与普通类的区别就是①不能创建实例对象,②允许有abstract方法。也可以这么理解——抽象类就是一个不能实例化的普通类,不过如果方法加了abstract,那么就必须在子类里面重写。
|
抽象类 |
接口 |
方法默认实现 |
支持 |
自Java8开始支持 |
实现 |
子类使用extends关键字来继承抽象类。子类如果不是抽象类,需要实现抽象类中声明的所有抽象方法 |
子类使用关键字implements来实现接口,需要实现接口中声明的所有方法 |
是否有构造函数 |
是 |
否 |
与正常Java类的区别 |
不能实例化抽象类,因为有abstract方法 |
接口是完全不同的类型 |
访问修饰符 |
public、protected和default |
只有public |
main方法 |
支持 |
不支持 |
多继承 |
继承一个类和实现多个接口 |
只可以继承一个或多个其它接口 |
速度 |
速度快 |
稍微有点慢,因为它需要时间去寻找在类中实现的方法 |
添加新方法 |
添加后可以给它提供默认的实现,故不需要改变现在的代码 |
自Java8开始,添加后不一定必须改变实现该接口的类,因为可以添加默认方法。 |
抽象类为什么不能实例化对象?
现实生活中也有抽象类的例子,比如说人类是一个抽象类,无法创建一个称作人类的对象,但是,人可以在继承人类后来创建对象。况且抽象类中的抽象方法只有声明,没有主体,如果实例化了,又如何去实现调用呢?
什么时候使用抽象类和接口?
•如果拥有一些方法并且想让它们中的一些有默认实现,那么使用抽象类吧。
•如果想实现多重继承,那么必须使用接口。由于Java不支持多继承,子类不能够继承多个类,但可以实现多个接口。因此就可以使用接口来解决它。
•如果基本功能在不断改变,那么就需要使用抽象类。如果不断改变基本功能并且使用接口,那么就需要改变所有实现了该接口的类。
下面接着再说说两者在应用上的区别。接口更多的是在系统架构设计方法发挥作用,主要用于定义模块之间的通信契约。而抽象类在代码实现方面发挥作用,可以实现代码的重用,例如,模板方法设计模式是抽象类的一个典型应用,假设某个项目的所有HTTP请求都要用相同的方式进行权限判断、访问日志记录和异常处理,那么就可以定义一个抽象的基类,让所有的controller都继承这个抽象基类,在抽象基类的service方法中实现上述功能,在各个子类中只是完成各自的业务逻辑代码,伪代码如下:
import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; public abstract class BaseServlet extends HttpServlet { public final void service(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { // 记录访问日志 // 进行权限判断 } protected abstract void doService(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException; // 注意访问权限定义成protected,显得既专业,又严谨,因为它是专门给子类用的 } class MyServlet1 extends BaseServlet { protected void doService(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { // 本Servlet只处理的具体业务逻辑代码 } }
父类方法中间的某段代码不确定,留给子类干,就用模板方法设计模式。
备注:这道题的思路是先从总体解释抽象类和接口的基本概念,然后再比较两者的语法细节,最后再说两者的应用区别。比较两者语法细节区别的条理是:先从一个类中的构造方法、普通成员变量和方法(包括抽象方法),静态变量和方法,继承性等6个方面逐一去比较回答,接着从第三者继承的角度的回答,特别是最后用了一个典型的例子来展现自己深厚的技术功底。