Java学习之抽象类和接口区别比较

2024-06-16 06:08

本文主要是介绍Java学习之抽象类和接口区别比较,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

抽象类(Abstract) && 接口(Interface)区别比较


第一个区别(变量&常量&修饰符)


新建抽象类AbstractTest:
   
public abstract class AbstractTest {
String name;
public String pub_name;
protected String prot_name;
private String pri_name;
static String sName;
public static String sPub_name;
protected static String sProt_name;
private static String sPri_name;
final String FNAME = "fname";
public final String FPUB_NAME = "fpub_name";
protected final String FPROT_NAME = "fprot_name";
private final String FPRI_NAME = "fpri_name";
static final String sFNAME = "fname";
public static final String sFPUB_NAME = "fpub_name";
protected static final String sFPROT_NAME = "fprot_name";
private static final String sFPRI_NAME = "fpri_name";
}

新建接口InterfaceTest:
   
public interface InterfaceTest {
 
String name;
public String pub_name;
protected String prot_name;
private String pri_name;
static String sName;
public static String sPub_name;
protected static String sProt_name;
private static String sPri_name;
final String FNAME = "fname";
public final String FPUB_NAME = "fpub_name";
protected final String FPROT_NAME = "fprot_name";
private final String FPRI_NAME = "fpri_name";
static final String sFNAME = "fname";
public static final String sFPUB_NAME = "fpub_name";
protected static final String sFPROT_NAME = "fprot_name";
private static final String sFPRI_NAME = "fpri_name";
}

下划线表示该处编译器提示报错,下面不再说明。

接口InterfaceTest中的错误处编译器都提示:InterfaceTest接口有 非法修饰的字段 ;只有public、static&final被允许作为修饰符。

修改接口InterfaceTest代码如下:
   
public interface InterfaceTest {
 
String name;
public String pub_name;
/* protected String prot_name;
private String pri_name;*/
static String sName;
public static String sPub_name;
/* protected static String sProt_name;
private static String sPri_name;*/
final String FNAME = "fname";
public final String FPUB_NAME = "fpub_name";
/* protected final String FPROT_NAME = "fprot_name";
private final String FPRI_NAME = "fpri_name";*/
static final String sFNAME = "fname";
public static final String sFPUB_NAME = "fpub_name";
/*protected static final String sFPROT_NAME = "fprot_name";
private static final String sFPRI_NAME = "fpri_name";*/
}

可以看到,注释了上面编译器报错的代码,但还是有提示报错:空白final字段 pub_name 尚未初始化。

由此可知, 接口的成员变量默认是final修饰的常量,是必须要初始化的。换句话说,接口不能有普通成员变量。

由以上代码的比较可以看出,抽象类与接口的第一个区别:

抽象类可以有成员变量、静态成员变量、常量和静态常量(常量必须被初始化),而且无访问权限限制;
接口不能有成员变量、静态成员变量,可以有由public、static&final修饰的常量。

第二个区别(构造方法)


抽象类AbstractTest添加如下代码;
   
public AbstractTest() {
}

接口也添加构造方法:
   
public InterfaceTest() {
}

编译器提示接口不能有构造方法。

由此得出抽象类与接口的第二个区别:

抽象类可以有构造方法,但接口不能有构造方法。

第三个区别(方法名&方法体&修饰符)


抽象类AbstractTest添加如下代码;
   
protected AbstractTest() {
}
protected void AbstractTest() {
}


可以看到, 方法名与抽象类名相同, 编译器没有报错,另外无论修改构造方法的访问级别为private还是默认级别都没报错。

接口InterfaceTest添加如下代码:
    
public void InterfaceTest() {
int a;
int b;
int c = a + b;
}

编译器提示: 抽象方法不用指定一个方法体。

修改后如下:
   
public void InterfaceTest();

这时编译器不报错了。而且 方法名与接口名相同。

修改InterfaceTest方法的访问权限:
   
protected void InterfaceTest();

编译器报错,提示:非法的修饰符, 接口方法的修饰符仅允许public和abstract。

由此得出,抽象类与接口的第三个区别:

抽象类和接口的方法名可以和类名可以相同,但抽象方法的访问权限没有限制,而接口方法的访问权限修饰符仅允许public和abstract,默认为public。
且抽象类中的方法可以有方法体(没有方法体的方法需要abstract修饰),而接口中的方法不能有方法体。

第四个区别(静态方法)


抽象类AbstractTest添加静态方法:
   
protected static void staticAbstractTest() {
}

接口也添加静态方法:
   
public static void InterfaceTest();

编译器报错,提示:接口中的方法有非法修饰符,仅允许public&abstract。

由此得出,抽象类和接口的第四个区别:

抽象类中可以包含静态方法,而接口中可以包含静态方法。

第五个区别(final方法)


抽象类AbstractTest添加final方法:
   
protected final void fAbstractTest() {
}

接口也添加final方法:
   
public final void InterfaceTest();

编译器报错,提示:接口中的方法有非法修饰符,仅允许public&abstract。

与第四个区别综合得出, 抽象类和接口的第五个区别:
抽象类中的普通方法可以被static和final修饰,而接口中的方法则不能被static和final修饰。

第六个区别(抽象方法)


由第二个到第五个区别可以得出如下结论:

抽象类中可以包含非抽象的普通方法,接口中的所有方法必须都是抽象的,不能有非抽象的普通方法。 

第七个区别(继承性)


分别copy抽象类和接口,粘贴其副本分别为AbstractTest2、InterfaceTest2,新建类InheritanceTest:

   
public class InheritanceTest extends AbstractTest, AbstractTest2 {
 
@Override
public void Abstract() {
// 继承自抽象类中的抽象方法
}
 
}

编译器报错,提示:语法错误。

修改如下:
   
public class InheritanceTest extends AbstractTest implements InterfaceTest, InterfaceTest2 {
 
@Override
public void Abstract() {
// 继承自抽象类中的抽象方法
}
 
}

编译通过。

由此得出抽象类和接口的第六个区别:

一个类可以实现多个接口,但只能继承一个抽象类。


如有疏漏或其他问题,欢迎指出互相交流,谢谢!




这篇关于Java学习之抽象类和接口区别比较的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1065652

相关文章

springboot集成easypoi导出word换行处理过程

《springboot集成easypoi导出word换行处理过程》SpringBoot集成Easypoi导出Word时,换行符n失效显示为空格,解决方法包括生成段落或替换模板中n为回车,同时需确... 目录项目场景问题描述解决方案第一种:生成段落的方式第二种:替换模板的情况,换行符替换成回车总结项目场景s

SpringBoot集成redisson实现延时队列教程

《SpringBoot集成redisson实现延时队列教程》文章介绍了使用Redisson实现延迟队列的完整步骤,包括依赖导入、Redis配置、工具类封装、业务枚举定义、执行器实现、Bean创建、消费... 目录1、先给项目导入Redisson依赖2、配置redis3、创建 RedissonConfig 配

SpringBoot中@Value注入静态变量方式

《SpringBoot中@Value注入静态变量方式》SpringBoot中静态变量无法直接用@Value注入,需通过setter方法,@Value(${})从属性文件获取值,@Value(#{})用... 目录项目场景解决方案注解说明1、@Value("${}")使用示例2、@Value("#{}"php

SpringBoot分段处理List集合多线程批量插入数据方式

《SpringBoot分段处理List集合多线程批量插入数据方式》文章介绍如何处理大数据量List批量插入数据库的优化方案:通过拆分List并分配独立线程处理,结合Spring线程池与异步方法提升效率... 目录项目场景解决方案1.实体类2.Mapper3.spring容器注入线程池bejsan对象4.创建

线上Java OOM问题定位与解决方案超详细解析

《线上JavaOOM问题定位与解决方案超详细解析》OOM是JVM抛出的错误,表示内存分配失败,:本文主要介绍线上JavaOOM问题定位与解决方案的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录一、OOM问题核心认知1.1 OOM定义与技术定位1.2 OOM常见类型及技术特征二、OOM问题定位工具

基于 Cursor 开发 Spring Boot 项目详细攻略

《基于Cursor开发SpringBoot项目详细攻略》Cursor是集成GPT4、Claude3.5等LLM的VSCode类AI编程工具,支持SpringBoot项目开发全流程,涵盖环境配... 目录cursor是什么?基于 Cursor 开发 Spring Boot 项目完整指南1. 环境准备2. 创建

Spring Security简介、使用与最佳实践

《SpringSecurity简介、使用与最佳实践》SpringSecurity是一个能够为基于Spring的企业应用系统提供声明式的安全访问控制解决方案的安全框架,本文给大家介绍SpringSec... 目录一、如何理解 Spring Security?—— 核心思想二、如何在 Java 项目中使用?——

SpringBoot+RustFS 实现文件切片极速上传的实例代码

《SpringBoot+RustFS实现文件切片极速上传的实例代码》本文介绍利用SpringBoot和RustFS构建高性能文件切片上传系统,实现大文件秒传、断点续传和分片上传等功能,具有一定的参考... 目录一、为什么选择 RustFS + SpringBoot?二、环境准备与部署2.1 安装 RustF

springboot中使用okhttp3的小结

《springboot中使用okhttp3的小结》OkHttp3是一个JavaHTTP客户端,可以处理各种请求类型,比如GET、POST、PUT等,并且支持高效的HTTP连接池、请求和响应缓存、以及异... 在 Spring Boot 项目中使用 OkHttp3 进行 HTTP 请求是一个高效且流行的方式。

java.sql.SQLTransientConnectionException连接超时异常原因及解决方案

《java.sql.SQLTransientConnectionException连接超时异常原因及解决方案》:本文主要介绍java.sql.SQLTransientConnectionExcep... 目录一、引言二、异常信息分析三、可能的原因3.1 连接池配置不合理3.2 数据库负载过高3.3 连接泄漏