Java中接口的存在有什么作用
知识点掌握了,还需要不断练习才能熟练运用。下面golang学习网给大家带来一个文章开发实战,手把手教大家学习《Java中接口的存在有什么作用》,在实现功能的过程中也带大家重新温习相关知识点,温故而知新,回头看看说不定又有不一样的感悟!
1. 什么是抽象类
在讲解接口之前,抽象类是绕不过去的一个概念,接口可以认为是一个比抽象类还要抽象的类。
什么是抽象类?「包含一个或多个抽象方法的类就是抽象类,抽象方法即没有方法体的方法」,抽象方法和抽象类都必须声明为 abstract
。例如:
<code>// 抽象类<br>public abstract class Person {<br> // 抽象方法<br> public abstract String getDescription();<br>}</code>
切记!「除了抽象方法之外,抽象类还可以包含具体数据和具体方法」。例如, 抽象类 Person
还保存着姓名和一个返回姓名的具体方法:
<code>public abstract class Person{<br> private String name;<br> public Person(String name){<br> this.name = name ;<br> }<br> public abstract String getDescription();<br> public String getName(){<br> return name;<br> }<br>}</code>
❝许多程序员都会「错误」的认为,在抽象类中不能包含具体方法。其实这也是接口和抽象类的不同之处,接口中是不能包含具体方法的。
❞
「抽象类不能被实例化」。也就是说,如果将一个类声明为 abstract
, 就不能创建这个类的对象。
<code>new Person("Jack"); // Error</code>
可以定义一个抽象类的对象变量, 但是它只能引用非抽象子类的对象。假设 Student
类是 Person
的非抽象子类:
<code>Person p = new Student("Jack"); // Right</code>
所谓非抽象子类就是说,如果创建一个继承抽象类的子类并为之创建对象,那么就「必须为父类的所有抽象方法提供方法定义」。如果不这么做(可以选择不做),子类仍然是一个抽象类,编译器会强制我们为新类加上 abstract
关键字。
下面定义扩展抽象类 Person
的具体子类 Student
:
<code>public class Student extends Person { <br> private String major; <br> public Student(String name, String major) { <br> super(name); <br> this.major = major; <br> } <br> @Override<br> public String getDescription(){ // 实现父类抽象方法<br> return "a student majoring in " + major; <br> } <br>} </code>
在 Student
类中实现了父类中的抽象方法 getDescription
。因此,「在 Student
类中的全部方法都是非抽象的, 这个类不再是抽象类」。
???? 调用如下:
<code>Person p = new Student("Jack","Computer Science");<br>p.getDescription();<br></code>
由于不能构造抽象类 Person
的对象, 所以变量 p
永远不会引用 Person
对象, 而是引用诸如 Student
这样的具体子类对象, 而这些对象中都重写了 getDescription
方法。
2. 什么是接口
接口的本质其实也是一个类,而且是一个比抽象类还要抽象的类。怎么说呢?抽象类是能够包含具体方法的,而接口杜绝了这个可能性,「在 Java 8 之前,接口非常纯粹,只能包含抽象方法,也就是没有方法体的方法」。而 Java 8 中接口出现了些许的变化,开始允许接口包含默认方法和静态方法,这个下文会讲解。
Java 使用关键字 interface
而不是 class
来创建接口。和类一样,通常我们会在关键字 interface
前加上 public
关键字,否则接口只有包访问权限,只能在接口相同的包下才能使用它。
<code>public interface Concept {<br> void idea1();<br> void idea2();<br>}</code>
同样的,接口中既然存在抽象方法,那么他就需要被扩展(继承)。使用 implements
关键字使一个类扩展某个特定接口(或一组接口),通俗来说:接口只是外形,现在这个扩展子类要说明它是如何工作的。
<code>class Implementation implements Concept {<br> @Override<br> public void idea1() {<br> System.out.println("idea1");<br> }<br> <br> @Override<br> public void idea2() {<br> System.out.println("idea2");<br> }<br>}<br></code>
这里需要注意的是,你可以选择显式地声明接口中的方法为 public
,但是「即使你不这么做,它们也是 public
的」。所以当实现一个接口时,来自接口中的方法必须被定义为 public
。否则,它们只有包访问权限,这样在被继承时,它们的可访问权限就被降低了,这是 Java 编译器所不允许的。
另外,接口中是允许出现常量的,与接口中的方法都自动地被设置为 public
—样,「接口中的域将被自动被设置为 public static final
类型」,例如:
<code>public interface Concept {<br> void idea1(); // public void idea1();<br> // 静态属性<br> double item = 95; // a public static final constant<br>}</code>
❝可以将接口方法标记为
❞public
,将域标记为public static final
。有些程序员出于习惯或提高清晰度的考虑, 愿意这样做。但 Java 语言规范却「建议不要书写这些多余的关键字」。
3. 接口的特性
接口和类其中不同的一点就是,我们「无法像类一样使用 new
运算符来实例化一个接口」:
<code>x = new Concept(. . .); // ERROR</code>
原因也很简单,接口连具体的构造方法都没有,肯定是无法实例化的。
当然, 尽管不能构造接口的对象,声明接口的变量还是可以的:
<code>Concept x; // OK</code>
接口变量必须引用实现了接口的类对象:
<code>x = new Implementation(. . .); // OK provided Implementation implements Concept</code>
接下来, 如同使用 instanceof
检查一个对象是否属于某个特定类一样, 也可以使用 instanceof
检查一个对象是否实现了某个特定的接口:
<code>if(x instanceof Concept){<br> ...<br>}</code>
另外,与可以建立类的继承关系一样,「接口也可以被继承」:
<code>public interface Concept1 {<br> void idea1();<br> void idea2();<br>}<br><br>-------------------------------------------<br> <br>public interface Concept2 extends Concept1{<br> double idea3();<br>}</code>
当然,读到这里大家可能依然无法理解,既然有了抽象类,为什么 Java 程序设计语言还要不辞辛苦地引入接口这个概念?
很重磅!因为「一个类可以实现多个接口,但是一个类只能继承一个父类」。正是接口的出现打破了 Java 这种单继承的局限,为定义类的行为提供了极大的灵活性。
<code>class Implementation implements Concept1, Concept2 // OK</code>
有一条实际经验:在合理的范围内尽可能地抽象。显然,接口比抽象类还要抽象。因此,一般更倾向使用接口而不是抽象类。
4. Java 8 接口新特性
上文提过一嘴,「在 Java 8 中,允许在接口中增加静态方法和默认方法」。理论上讲,没有任何理由认为这是不合法的,只是这有违于将接口作为抽象规范的初衷。举个例子:
<code>public interface Concept {<br> // 静态方法<br> public static void get(String name){<br> System.out.println("hello " + name);<br> }<br> // 默认方法<br> default void idea1(){<br> System.out.println("this is idea1");<br> };<br>}</code>
用 default
修饰符标记的方法就是默认方法,这样子类就不需要去实现这个方法了。
不过,引入默认方法后,就出现了一个「默认方法冲突」的问题。如果先在一个接口 A 中将一个方法 idea
定义为默认方法, 然后又在另一个接口 B 或者超类 C 中定义了同样的方法 idea
,然后类 D 实现了这两个接口 A 和 B(或超类 C)。于是类 D 中就有了方法 idea
的两个默认实现,出现了冲突,为此,Java 制定了一套规则来解决这个二义性问题:
1 ) 「超类优先」。如果超类提供了一个具体方法,接口中的同名且有相同参数类型的默认方法会被忽略。
2 ) 「接口冲突」。如果一个父类接口提供了一个默认方法,另一个父类接口也提供了一个同名而且参数类型相同的方法,子类必须覆盖这个方法来解决冲突。例如:
<code>interface A {<br> default void idea(){<br> System.out.println("this is A");<br> }<br>}<br><br>interface B {<br> default void idea(){<br> System.out.println("this is B");<br> }<br>}<br><br>// 需要在 D 类中覆盖 idea 方法<br>class D implements A, B{<br> public void getName(){<br> System.out.println("this is D");<br> }<br>}</code>
现在假设 B
接口没有为 idea
提供默认实现:
<code>interface B {<br> void idea();<br>}</code>
那么 D 类会直接从 A 接口继承默认方法吗?这好像挺有道理, 不过,Java 设计者更强调一致性。两个接口如何冲突并不重要,「只要有一个接口提供了一个默认实现,编译器就会报告错误, 我们就必须解决这个二义性」。
当然,如果两个接口都没有为共享方法提供默认实现, 那么就与 Java 8 之前的情况一样,这里不存在冲突。
5. 接口存在的意义
在我自己早期学习编程的时候,对接口存在的意义实在困惑,我自己乱写代码的时候基本上不可能意识到需要去写接口,不知道接口到底有什么用,为什么要定义接口,感觉定义接口只是提前做了个多余的工作。
其实不是,定义接口并非多余,「接口是用来提供公用的方法,规定子类的行为的」。举个例子,让大家直观的感受下接口的作用:
比如有个网站, 需要保存不同客户的信息, 有些客户从 Web 网站来, 有些客户从手机客户端来, 有些客户直接从后台管理系统录入。假设不同来源的客户有不同的处理业务流程, 这个时候我们定义接口来提供一个保存客户信息的方法,然后不同的平台实现我们这个保存客户信息的接口,以后保存客户信息的话, 我们只需要知道这个接口就可以了,具体调用的方法被封装成了黑盒子,这也就是 Java 的多态的体现,「接口帮助我们对这些有相同功能的方法做了统一管理」。
本篇关于《Java中接口的存在有什么作用》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于文章的相关知识,请关注golang学习网公众号!

- 上一篇
- Java开发实战:利用七牛云CDN加速网站访问

- 下一篇
- 实用技巧:如何使用Java调用七牛云云市场接口实现订购服务
-
- 文章 · java教程 | 13分钟前 | 日期格式化 java8 LocalDate DateTimeFormatter 日期操作
- JavaLocalDate从入门到精通,手把手教你玩转Java8日期类
- 284浏览 收藏
-
- 文章 · java教程 | 46分钟前 | prometheus grafana Micrometer 指标 Java应用监控
- Java如何用Prometheus实现监控?保姆级教程
- 296浏览 收藏
-
- 文章 · java教程 | 1小时前 | java 并发编程 线程同步 countdownlatch 倒计时门闩
- Java进阶之路!CountDownLatch倒计时门闩实战详解
- 357浏览 收藏
-
- 文章 · java教程 | 2小时前 | 线程池 异常处理 异步编程 completablefuture 回调机制
- 手把手教你用JavaCompletableFuture玩转异步编程
- 209浏览 收藏
-
- 文章 · java教程 | 2小时前 | 反序列化 Java序列化 transient关键字 敏感信息保护 Externalizable接口
- Java中transient关键字的用法及常见应用场景
- 162浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- Java菜鸟必看!类的定义+作用,一文全搞懂!
- 291浏览 收藏
-
- 文章 · java教程 | 4小时前 | 空指针异常 空值处理 链式操作 JavaOptional orElse
- JavaOptional避坑指南:4大实用技巧搞定空值处理
- 390浏览 收藏
-
- 文章 · java教程 | 4小时前 |
- Java中类和对象傻傻分不清?区别联系一文搞定!
- 245浏览 收藏
-
- 文章 · java教程 | 5小时前 | 任务队列 Java线程池 threadpoolexecutor Executors 线程数
- Java线程池这样用才爽!四大线程池场景实战全解析
- 186浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- 茅茅虫AIGC检测
- 茅茅虫AIGC检测,湖南茅茅虫科技有限公司倾力打造,运用NLP技术精准识别AI生成文本,提供论文、专著等学术文本的AIGC检测服务。支持多种格式,生成可视化报告,保障您的学术诚信和内容质量。
- 60次使用
-
- 赛林匹克平台(Challympics)
- 探索赛林匹克平台Challympics,一个聚焦人工智能、算力算法、量子计算等前沿技术的赛事聚合平台。连接产学研用,助力科技创新与产业升级。
- 78次使用
-
- 笔格AIPPT
- SEO 笔格AIPPT是135编辑器推出的AI智能PPT制作平台,依托DeepSeek大模型,实现智能大纲生成、一键PPT生成、AI文字优化、图像生成等功能。免费试用,提升PPT制作效率,适用于商务演示、教育培训等多种场景。
- 87次使用
-
- 稿定PPT
- 告别PPT制作难题!稿定PPT提供海量模板、AI智能生成、在线协作,助您轻松制作专业演示文稿。职场办公、教育学习、企业服务全覆盖,降本增效,释放创意!
- 81次使用
-
- Suno苏诺中文版
- 探索Suno苏诺中文版,一款颠覆传统音乐创作的AI平台。无需专业技能,轻松创作个性化音乐。智能词曲生成、风格迁移、海量音效,释放您的音乐灵感!
- 85次使用
-
- 提升Java功能开发效率的有力工具:微服务架构
- 2023-10-06 501浏览
-
- 掌握Java海康SDK二次开发的必备技巧
- 2023-10-01 501浏览
-
- 如何使用java实现桶排序算法
- 2023-10-03 501浏览
-
- Java开发实战经验:如何优化开发逻辑
- 2023-10-31 501浏览
-
- 如何使用Java中的Math.max()方法比较两个数的大小?
- 2023-11-18 501浏览