需求:接口是否可继承接口?抽象类是否可实现(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,那么就必须在子类里面重写。
抽象类 | 接口 | |
方法默认实现 | 支持 | 不支持,接口完全是抽象的 |
实现 | 子类使用extends关键字来继承抽象类。子类如果不是抽象类,需要实现抽象类中声明的所有抽象方法 | 子类使用关键字implements来实现接口,需要实现接口中声明的所有方法 |
是否有构造函数 | 是 | 否 |
与正常Java类的区别 | 不能实例化抽象类,因为有abstract方法 | 接口是完全不同的类型 |
访问修饰符 | public、protected和default | 只有public |
main方法 | 支持 | 不支持 |
多继承 | 继承一个类和实现多个接口 | 只可以继承一个或多个其它接口 |
速度 | 速度快 | 稍微有点慢,因为它需要时间去寻找在类中实现的方法 |
添加新方法 | 添加后可以给它提供默认的实现,故不需要改变现在的代码 | 添加后必须改变实现该接口的类 |
抽象类为什么不能实例化对象?
现实生活中也有抽象类的例子,比如说人类是一个抽象类,无法创建一个称作人类的对象,但是,人可以在继承人类后来创建对象。况且抽象类中的抽象方法只有声明,没有主体,如果实例化了,又如何去实现调用呢?
什么时候使用抽象类和接口?
- 如果拥有一些方法并且想让它们中的一些有默认实现,那么使用抽象类吧。
- 如果想实现多重继承,那么必须使用接口。由于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个方面逐一去比较回答,接着从第三者继承的角度的回答,特别是最后用了一个典型的例子来展现自己深厚的技术功底。
到此这篇关于Java中接口和抽象类的区别详解的文章就介绍到这了,更多相关Java 接口 抽象类内容请搜索自学编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持自学编程网!
- 本文固定链接: https://www.zxbcw.cn/post/191573/
- 转载请注明:必须在正文中标注并保留原文链接
- QQ群: PHP高手阵营官方总群(344148542)
- QQ群: Yii2.0开发(304864863)