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 {@Overridepublic void walk() {// 打印接口中的常量System.out.println(Person.NAME);System.out.println("学生可以走路");}@Overridepublic 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 {@Overridepublic void abstractMethod2() {...}@Overridepublic 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 {@Overridepublic void run() {System.out.println("学生可以跑步");}
}

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

public class Student implements Person {@Overridepublic void run() {System.out.println("学生可以跑步");}// 重写默认方法@Overridepublic 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 {@Overridepublic 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 开始,接口中可以定义默认方法和静态方法。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

PostgreSQL事务基础理解

PostgreSQL事务 事务是数据库管理系统执行过程中的一个逻辑单位,由一个有限的数据库操作序列构成。数据库事务通常包含一个序列对数据库的读和写操作,主要是包含以下两个目的: 为数据库操作序列提供一个从失败中恢复到正常状态的方法&#…

事务管理控制

文章目录 1. 事务的基本概念2. 数据库的并发控制2.1 事务调度2.2 并发操作带来的问题2.3 并发调度的可串行性2.4 并发控制技术2.5 两段锁协议2.6 多粒度封锁协议 3. 数据库的备份与恢复3.1 数据库系统故障3.2 数据库的备份3.3 数据库的恢复 4. 数据库的安全性与完整性4.1 数据库…

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

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

网络变压器原理与维修视频

如果你在寻找网络变压器原理与维修方面的视频教程,可以在网上找一些优质的资料进行学习。一些国内外知名的教育和科技资源网站,如B站、优酷、YouTube等,都有涉及网络变压器原理和维修的视频教程,你可以在这些网站上进行搜索。 通常…

智慧之选:开源与闭源大模型的未来探索

✨✨ 欢迎大家来访Srlua的博文(づ ̄3 ̄)づ╭❤~✨✨ 🌟🌟 欢迎各位亲爱的读者,感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua小谢,在这里我会分享我的知识和经验。&am…

淘宝订单系统ERP中如何接入平台订单信息?(订单API)

淘宝开放平台中有交易API,里面有各种关于交易的API接口。但是申报应用权限的审核流程严格又漫长。不少公司费时费力的申请后,结果还是没有审批下来。 调用淘宝自定义接口custom,可以实现淘宝开放平台API的调用。技术人员会根据您需要的接口做…

基于SpringBoot的网盘系统设计与实现

第1章 绪论... 1 1.1 研究背景与意义... 1 1.1.1 研究背景... 1 1.1.1 研究意义... 1 1.2 国内外研究现状... 2 1.2.1 国内研究现状... 2 1.2.2 国外研究现状... 3 1.3 论文组织架构... 4 第2章 关键技术介绍... 5 2.1 SpringBoot. 5 2.2 MySQL数据库... 5 2.3 MVC架…

Java进阶学习笔记2——static修饰成员变量

static: 叫静态,可以修饰成员变量、成员方法。 成员变量按照有无static修饰,分为两种: 类变量:有static修饰,属于类,在计算机中只有一份,会被类的全部对象共享。静态成员变量。 实…

【LeetCode】30.串联所有单词的子串

串联所有单词的子串 题目描述: 给定一个字符串 s 和一个字符串数组 words。 words 中所有字符串 长度相同。 s 中的 串联子串 是指一个包含 words 中所有字符串以任意顺序排列连接起来的子串。 例如,如果 words ["ab","cd",&qu…

基金/证券项目如何进行非交易日数据补全(实战)

一些大数据开发的项目,特别是基金/证券公司的项目,都经常会涉及到交易日与非交易日的概念。 如果要让你对一张交易日跑批的主表,怎么去补全非交易日的数据呢? 在遇到这种情况的时候,我们要去怎么处理?来&…

webSocket+Node+Js实现在线聊天(包含所有代码)

这篇文章主要介绍了如何使用 webSocket、Node 和 Js 实现在线聊天功能。 重要亮点 💻 技术选型:使用 Node.js 搭建服务器,利用 Express 框架和 Socket.io 库实现 WebSocket 通信。 📄 实现思路:通过建立数组存储聊天…

掌握RESTful API:从入门到精通,全面解析Web开发的基石!

在现代Web开发中,API(应用程序编程接口)已经成为不同系统之间通信的重要手段。其中,RESTful API是一种基于HTTP协议的设计风格,它简洁、易用且高效。作为一个资深的技术人员,本文将全面详细地介绍RESTful A…

等保建设:打造MySQL数据库审计系统

1、建设目标 在等级保护三级->应用安全->安全审计中强制需要有审计平台(满足对操作系统、数据库、网络设备的审计,在条件不允许的情况下,至少要使用数据库审计) 数据库审计服务符合等级保护三级标准,帮助您满足合规性要求,…

VsCode CMake调试QT无法查看源码问题处理

遇到的问题 当我们在VsCode使用CMake来调试QT程序时,想F11进入到QT源码时,发现进不去,无法查看源码。 原因 这种情况一般都是安装目录下没有pdb文件导致的。 PDB文件:是一个包含调试信息的数据库,它由编译器和链接器…

用UDP写一个回显服务器和一个字典服务器

回显服务器 操作系统提供了一些网络通信的api(socket)。 如: 回显服务器:请求是啥?响应就是啥。 一个正常的服务器要做三件事情: 1. 读取请求并解析。 2. 根据请求计算响应。 3. 把响应写回给客户端。…

【ETAS CP AUTOSAR工具链】ARXML文件详解

本篇文章首先对ARXML这种文件格式做了一个概述,叙述了这种标签语言的基本语法(如果您用HTML做过网页,那么这种格式您一定不会陌生),然后对ARXML文件都会包含的一些基本信息做了详细的解读,最后基于使用ISOL…

Android:使用Kotlin搭建MVP架构模式

一、简介Android MVP架构模式 MVP全称:Model、View、Presenter; View:负责视图部分展示Model:负责数据的请求、解析、过滤等数据层操作。Presenter:View和Model交互的桥梁。对应MVC中的C(controller&#x…

01.爬虫---初识网络爬虫

01.初识网络爬虫 1.什么是网络爬虫2.网络爬虫的类型3.网络爬虫的工作原理4.网络爬虫的应用场景5.网络爬虫的挑战与应对策略6.爬虫的合法性总结 1.什么是网络爬虫 网络爬虫,亦称网络蜘蛛或网络机器人,是一种能够自动地、系统地浏览和收集互联网上信息的程…

路由聚合和VRRP技术

实验拓扑图: 实验需求 1、内网IP地址使用172.16.0.0/16 2、SW1和SW2之间互为备份; 3、VRRP/stp/vlan/eth-trunk均使用; 4、所有pc均通过DHCP获取IP地址; 5、ISP只配置IP地址; 6、所有电脑可以正常访问ISP路由器环…

【学习笔记】Windows GDI绘图(五)图形路径GraphicsPath详解(上)

文章目录 图形路径GraphicsPath填充模式FillMode构造函数GraphicsPath()GraphicsPath(FillMode)GraphicsPath(Point[],Byte[])和GraphicsPath(PointF[], Byte[])GraphicsPath(Point[], Byte[], FillMode)和GraphicsPath(PointF[], Byte[], FillMode)PathPointType 属性FillMode…