23. 【Java教程】接口

本小节我们将学习 Java 接口(interface),通过本小节的学习,你将了解到什么是接口、为什么需要接口、如何定义和实现接口,以及接口的特点等内容。最后我们也将对比抽象类和接口的区别。

1. 概念

Java 接口是一系列方法的声明,是一些方法特征的集合,一个接口只有方法的特征没有方法的实现。

在 Java 中,被关键字 interface 修饰的 class 就是一个接口。接口定义了一个行为协议,可以由类层次结构中任何位置的任何类实现。接口中定义了一组抽象方法,都没有具体实现,实现该接口的类必须实现该接口中定义的所有抽象方法。

2. 为什么需要接口

我们知道 Java 仅支持单继承,也就是说一个类只允许有一个直接父类,这样保证了数据的安全。Java 不支持下图所示的多继承:

接口就是为了解决 Java 单继承这个弊端而产生的,虽然一个类只能有一个直接父类,但是它可以实现多个接口,没有继承关系的类也可以实现相同的接口。继承和接口的双重设计既保持了类的数据安全也变相实现了多继承。

3. 接口的定义和实现

3.1 定义接口

3.1.1 接口声明

使用 interface 关键字声明一个接口:

public interface Person {
    ...
}

接口声明需要两个元素:interface 关键字和接口名称,public 修饰符表示该接口可以在任何包的任何类中使用,如果为显示指定访问修饰符,则该接口只能被在同包中的类使用。

3.1.2 接口主体

接口主体中,可以定义常量和方法声明:

public interface Person {
  	final String NAME = "我是Person接口中的常量";
	void walk();
  	void run();
}

上面的 Person 就是一个接口,这个接口定义了一个常量 NAME 和两个抽象方法 walk()run()

接口比抽象类更加 “抽象”,它下面不能拥有具体实现的方法,必须全部都是抽象方法,所有的方法默认都是 public abstract 的,所以在接口主体中的方法,这两个修饰符无需显示指定。

接口除了方法声明外,还可以包含常量声明。在接口中定义的所有的常量默认都是 publicstatic,和 final 的。

接口中的成员声明不允许使用 private 和 protected 修饰符。

3.2 实现接口

接口定义了一些行为协议,而实现接口的类要遵循这些协议。implements 关键字用于实现接口,一个类可以实现一个或多个接口,当要实现多个接口时,implements 关键字后面是该类要实现的以逗号分割的接口名列表。其语法为:

public class MyClass implements MyInterface1, MyInterface2 {
   ...
}

下面是实现了 Person 接口的 Student 类的示例代码:

public class Student implements Person {
    @Override
    public void walk() {
      	// 打印接口中的常量
        System.out.println(Person.NAME);
        System.out.println("学生可以走路");
    }

    @Override
    public void run() {
        System.out.println("学生可以跑步");
    }
}

上述代码中,Student 类实现了 Person 接口。值得注意的是,可以使用接口名。常量名的方式调用接口中所声明的常量:

String name = Person.NAME;

4. 接口继承

接口也是存在继承关系的。接口继承使用 extends 关键字。例如,声明两个接口 MyInterface1 和 MyInterface2MyInterface2 继承自 MyInterface1

// MyInterface1.java
public interface MyInterface1 {
    void abstractMethod1();
}

// MyInterface2.java
public interface MyInterface2 extends MyInterface1 {
    void abstractMethod2();
}

当一个类实现 MyInterface2 接口,将会实现该接口所继承的所有抽象方法:

// MyClass.java
public class MyClass implements MyInterface2 {
    @Override
    public void abstractMethod2() {
				...
    }

    @Override
    public void abstractMethod1() {
				...
    }
}

值得注意的是,一个接口可以继承多个父接口,接口名放在 extends 后面,以逗号分割,例如:

// MyInterface1.java
public interface MyInterface1 {
    void abstractMethod1();
}

// MyInterface2.java
public interface MyInterface2 {
    void abstractMethod2();
}

// MyInterface3.java
public interface MyInterface3 extends MyInterface1, MyInterface2 {
    void abstractMethod3();
}

补充一点,当一个实现类存在 extends 关键字,那么 implements 关键字应该放在其后:

public class MyClass extends SuperClass implements MyInterface {
   ...
}

5. 默认方法和静态方法

从 JDK 1.8 开始,接口中可以定义默认方法和静态方法。与抽象方法不同,实现类可以不实现默认方法和类方法。

5.1 默认方法

5.1.1 声明

我们可以使用 default 关键字,在接口主题中实现带方法体的方法,例如:

public interface Person {
  	void run();
  
  	default void eat() {
      	System.out.println("我是默认的吃方法");
    }
}
5.1.2 调用和重写

在实现类中,可以不实现默认方法:

public class Student implements Person {
  	@Override
    public void run() {
        System.out.println("学生可以跑步");
    }
}

我们也可以在实现类中重写默认方法,重写不需要 default 关键字:

public class Student implements Person {
  	@Override
    public void run() {
        System.out.println("学生可以跑步");
    }
  
  	// 重写默认方法
  	@Override
  	public void eat() {
      	// 使用 接口名.super.方法名() 的方式调用接口中默认方法
      	Person.super.eat();
      	System.out.println("学生吃东西");
    }
}

如果想要在实现类中调用接口的默认方法,可以使用接口名.super. 方法名 () 的方式调用。这里的 接口名.super 就是接口的引用。

5.1.3 使用场景

当一个方法不需要所有实现类都进行实现,可以在接口中声明该方法为默认方法;使用默认方法还有一个好处,当接口新增方法时,将方法设定为默认方法,只在需要实现该方法的类中重写它,而不需要在所有实现类中实现。

5.2 静态方法

5.2.1 声明

使用 static 关键字在接口中声明静态方法,例如:

public interface Person {
    void walk();
    // 声明静态方法
    static void sayHello() {
        System.out.println("Hello mybj!");
    }
}
5.2.2 调用

类中的静态方法只能被子类继承而不能被重写,同样在实现类中,静态方法不能被重写。如果想要调用接口中的静态方法,只需使用 接口名。类方法名 的方式即可调用:

public class Student implements Person {
    @Override
    public void walk() {
      	// 调用接口中的类方法
        Person.sayHello();
        System.out.println("学生会走路");
    }
}

6. 接口和抽象类的区别

  1. 接口的方法默认是 public ,所有方法在接口中不能有实现(Java 8 开始接口方法可以有默认实现),而抽象类可以有非抽象的方法;
  2. 接口中除了 static 、final 变量,不能有其他变量,而抽象类可以;
  3. 一个类可以实现多个接口,但只能实现一个抽象类。接口自己本身可以通过 extends 关键字扩展多个接口;
  4. 接口方法默认修饰符是 public ,抽象方法可以有 public 、protected 和 default 这些修饰符(抽象方法就是为了被重写所以不能使用 private 关键字修饰!);
  5. 从设计层面来说,抽象是对类的抽象,是一种模板设计,而接口是对行为的抽象,是一种行为的规范。

7. 多个接口中的重名成员解决方法

7.1 多个接口存在重名默认方法

例如有两个接口 MyInteface1.java 和 MyInterface2.java,存在相同签名的默认方法:

public interface MyInterface1 {
    default void defaultMethod() {
        System.out.println("我是MyInterface1接口中的默认方法");
    }
}

public interface MyInterface2 {
    default void defaultMethod() {
        System.out.println("我是MyInterface2接口中的默认方法");
    }
}

当实现类实现两个接口时,同名的默认方法将会发生冲突,解决办法是在实现类中重写这个默认方法

public class MyClass implements MyInterface1, MyInterface2 {
	public void defaultMethod() {
      	System.out.println("我是重写的默认方法");
    }
}

还有一种情况:实现类所继承的父类中也存在与默认方法的同名方法,此时存在三个同名方法:

// 声明父类,并在父类中也定义同名方法
public class SuperClass {
  	public void defaultMethod() {
        System.out.println("我是SuperClass中的defaultMethod()方法");
    }
}

// 实现类继承父类,并实现两个接口
public class MyClass extends SuperClass implements MyInterface1, MyInterface2 {
}

实例化 MyClass 类,调用其 defaultMethod() 方法:

MyClass myClass = new MyClass();
myClass.defaultMethod();

此时编译执行,不会报错:

我是SuperClass中的defaultMethod()方法

实际上,在没有重写的情况下,它执行了实现类的父类 SuperClass 的 defaultMethod() 方法。

7.2 多个接口中存在重名常量

例如有两个接口,存在重名的常量:

public interface MyInterface1 {
    final int NUM = 100;
}

public interface MyInterface2 {
	final int NUM = 200;
}

此时在实现类中,我们可以使用接口名。常量名的方式分别调用:

public MyClass implements MyInterface1, MyInterface2 {
	  System.out.println(MyInterface1.NUM);  	
	  System.out.println(MyInterface2.NUM);  	
}

当实现类将入一个继承关系时:

class SuperClass {
  	static int NUM = 300;
}

public MyClass extends SuperClass implements MyInterface1, MyInterface2 {	
    System.out.println(NUM);
}

当父类中的属性或常量与接口中的常量同名时,子类无法分辨同名的 NUM 是哪一个。编译程序将会报错:

MyClass.java:4: 错误: 对NUM的引用不明确
        System.out.println(NUM);
                           ^
  SuperClass 中的变量 NUM 和 MyInterface1 中的变量 NUM 都匹配
1 个错误

此时只有在子类中声明 NUM,才可以通过编译:

public MyClass extends SuperClass implements MyInterface1, MyInterface2 {
  	int NUM = 3;
	System.out.println(NUM);
}

8. 小结

通过本小节的学习,我们知道了 Java 的接口是为了解决其单继承的弊端而产生的,可以使用 interface 关键字来声明一个接口,接口内部不能有具体的方法实现。可以使用 implements 关键字来实现接口,一个接口可以继承多个父接口,接口名放在 extends 后面,以逗号分割。从 Java 8 开始,接口中可以定义默认方法和静态方法。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/640451.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【教程】Linux 安装 kkFileView 文档在线预览项目 及优化

【教程】Linux 安装 kkFileView 文档在线预览项目 官网 kkFileView - 在线文件预览 (keking.cn) 安装包 可以直接下载成品 也可以下载source 源码 自己编译 kkFileView 发行版 - Gitee.com 打开IDEA 然后先clear 再install 然后在 file-online-preview\server\target 目录…

YOLOv10论文解读:实时端到端的目标检测模型

《博主简介》 小伙伴们好,我是阿旭。专注于人工智能、AIGC、python、计算机视觉相关分享研究。 ✌更多学习资源,可关注公-仲-hao:【阿旭算法与机器学习】,共同学习交流~ 👍感谢小伙伴们点赞、关注! 《------往期经典推…

Day 56 647. 回文子串 516.最长回文子序列

回文子串 给定一个字符串,你的任务是计算这个字符串中有多少个回文子串。 具有不同开始位置或结束位置的子串,即使是由相同的字符组成,也会被视作不同的子串。 示例 1: 输入:“abc”输出:3解释&#xf…

【LeetCode】【5】最长回文子串

文章目录 [toc]题目描述样例输入输出与解释样例1样例2 提示Python实现动态规划 个人主页:丷从心 系列专栏:LeetCode 刷题指南:LeetCode刷题指南 题目描述 给一个字符串s,找到s中最长的回文子串 样例输入输出与解释 样例1 输入…

概念艺术3D三维虚拟展览系统让更多人一同领略艺术的无穷魅力

经过多年的技术积累,华锐视点3D云展平台为各位提供的网上3D书画展厅,是一个集逼真视觉体验与沉浸式感官享受于一体的线上艺术殿堂。通过先进的Web3D实时渲染技术,打造全景3D立体场景,让您仿佛置身于实体展厅之中,感受那…

美业系统源码美业SaaS系统-门店卡项已线下退款,需要作废怎么处理?

美业SaaS系统源码 连锁门店美业收银系统源码 收银管理 / 会员管理 / 预约管理 / 排班管理 / 商品管理 / 活动促销 PC管理后台、手机APP、iPad APP、微信小程序 1、加盟店卡项线下退款处理方法: 询问具体退款会员手机号和卡项,找到需要退款的订单号。…

Spark中RDD概述及RDD算子详解

一、RDD概述 1、RDD: 弹性的分布式数据集 弹性:RDD 中的数据即可以缓存在内存中, 也可以缓存在磁盘中, 也可以缓存在外部存储中 分布式:数据可以分布在多台服务器中,RDD中的分区来自于block块,而block块会来自不同的datanode 数…

华为数通 HCIP-Datacom(H12-821)题库

最新 HCIP-Datacom(H12-821)完整题库请扫描上方二维码访问,持续更新中。 BGP路由的Update消息中可不包含以下哪些属性? A、Local Preference B、AS Path C、MED D、Origin 答案:AC 解析:as-path和ori…

VMware虚拟机桥接无线网卡上网(WIFI)

一、打开VM点击【编辑】-【虚拟网络编辑器】 二、点击【桥接模式】- 点击【自动设置】- 选择自己的无线网适配器 - 【确定】 三、开机之后会弹出提示连接网络,就能看见网络已经连上了

图片分类模型训练及Web端可视化预测(下)——Web端实现可视化预测

Web端实现可视化预测 基于Flask搭建Web框架,实现HTML登录页面,编写图片上传并预测展示页面。后端实现上一篇文章所训练好的模型,进行前后端交互,选择一张图片,并将预测结果展示到页面上。 文章目录 Web端实现可视化预测…

Apache Flink CDC 3.1.0版本知识学习

Apache Flink CDC 3.1.0版本知识学习 一、Flink CDC 3.1 快速预览二、Transformation 支持三、分库分表合并支持四、使用 Kafka Pipeline Sink 高效写入 Canal/Debezium 格式数据五、更高效地实时入湖 Paimon六、其他改进七、Flink CDC 3.1 版本兼容性 一、Flink CDC 3.1 快速预…

[深入理解DDR5] 2-1 封装与引脚

3500字,依公知及经验整理,原创保护,禁止转载。 专栏 《深入理解DDR》 1 DDR5 颗粒 X4 X8 X16 这里的 X8 or X16, 可以理解为一个DRAM芯片有几个存储阵列。“X几”。进行列寻址时会同时从几个阵列的同一个坐标位置读出数据bit来&a…

博客系统(Servlet实现)

目录 1.准备工作 2.数据库设计 2.1表设计 2.2封装数据库操作代码 2.3创建 Blog 类 和 User 类 2.4创建 BlogDao 类和 UserDao 类 3.读取博客列表功能 3.1约定前后端交互接口 3.2实现服务器代码 3.3实现客户端代码 4.实现博客详情 4.1约定前后端交互接口 4.2实现服…

网站流量统计分析

网站流量统计分析:洞悉用户行为的关键 在当今数字化时代,网站流量统计分析已经成为了企业成功的关键因素之一。通过深入了解用户的行为和偏好,企业可以更好地调整其营销策略、优化用户体验以及提高转化率。本文将探讨网站流量统计分析的重要性…

av_dump_format经验分析,FFmpeg获取媒体文件总时长(FLV获取总时长的误区)

播放器有个功能,当用户打开视频时,需要读取媒体文件的总时长等信息,不巧的时,获取FLV时总失败,下面来具体分析下FLV和MP4获取总时长的原因和区别: 播放器有个获取MediaInfo的接口,功能如下&am…

【面试干货】矩阵对角线元素之和

【面试干货】矩阵对角线元素之和 1、实现思想2、代码实现 💖The Begin💖点点关注,收藏不迷路💖 1、实现思想 创建一个3x3的二维数组来表示输入的矩阵。通过嵌套循环读取输入的矩阵元素,并将其保存到数组中。再次嵌套循…

Linux基础入门和帮助-第二篇

马哥教育 Linux SRE 学习笔记 用户登录信息查看命令 whoami: 显示当前登录有效用户 [rootrocky8 ~]$whoami rootwho: 系统当前所有的登录会话 [rootrocky8 ~]$who root pts/0 2024-05-24 12:55 (10.0.0.1)w: 系统当前所有的登录会话及所做的操作 [rootrocky8 ~]…

Springboot开发 -- Postman 使用指南

引言 在 Spring Boot 应用开发过程中,接口测试是必不可少的一环。Postman 作为一款强大的 API 开发和测试工具,可以帮助开发者轻松构建、测试和管理 HTTP 请求。本文将为大家介绍如何在 Spring Boot 开发中使用 Postman 进行接口测试。 一、准备工作 安…

【算法】分治 - 快速排序

快乐的流畅:个人主页 个人专栏:《算法神殿》《数据结构世界》《进击的C》 远方有一堆篝火,在为久候之人燃烧! 文章目录 引言一、颜色分类二、排序数组三、数组中的第k个数四、最小的k个数总结 引言 本节主要介绍快速排序&#xf…