假面与演员:到底是接口在使用类,还是类在使用接口?编程接口与物理接口的区别又是什么?
前言:本篇文章解释了接口学习过程中的2个常见问题,一个是“为什么是类在使用接口”,另一个一个是“编程接口与物理接口的差异源于所处的抽象层次和交互模式的不同”,旨在揭示编程接口的本质。
Part1.是类在使用接口
当学习接口时,很多人都会告诉你“是类在使用接口”。但是我们观察接口的运用方式,会发现我们常常是用接口类型的变量来接收类的实例,并使用类中的具体方法。这不禁让我们产生疑问:到底是接口在使用类,还是类在使用接口?
先说结论:“是类在使用接口”,而不是接口在使用类。下面由我来为大家揭示控制反转的微妙之处。
一. 表象:为什么“看起来像接口在使用类”?
1. 代码的直观表现
当用接口类型的变量调用方法时,代码的书写形式确实像是“接口在操作类”:
// 接口类型变量
Flyable obj = new Bird();
obj.fly(); // 看似接口调用了 Bird 的方法
这里 Flyable 类型的变量 obj 调用了 fly() 方法,而实际执行的是 Bird 类的实现。这种写法容易让人产生“接口主动使用类”的错觉。
2. 直觉误区来源
-
语法焦点:代码的书写顺序(先声明接口变量,再调用方法)暗示了“接口是主语”。
-
隐藏动态绑定:编译时只知道
obj是Flyable类型,但运行时实际调用哪个类的fly()是动态决定的。这种“延迟绑定”掩盖了类的主动性。
二. 本质:假面与演员。。
1. 接口是“面具”,类是“演员”
接口类型的变量更像是一个角色标签。它不关心背后具体是哪个类,只关心这个类是否“戴上了面具”(实现了接口)。观众们(我们)看到并记住的是这个假面(接口),真正在表演的是演员(类)。
真正的执行者是类自身的方法:
// 逻辑拆解:
Flyable obj = new Bird(); // 类主动“戴上面具”(实现接口)
obj.fly(); // 面具下的演员(Bird)在表演
2. 接口的“被动性”
在运行时,JVM/编译器会根据对象实际类型调用对应方法,与接口无关。
接口没有能力“主动使用”任何类,它只是:
-
定义规范:列出需要实现的方法清单。
-
提供类型标识:允许其他代码以统一的方式操作不同类。
三. 纠正思维:重新理解主动权!
1. 类才是行为的拥有者
接口只是强制类公开某些能力,但能力的具体实现完全由类决定。
例如:
interface Weapon {void attack();
}class Sword implements Weapon {public void attack() { System.out.println("挥剑"); } // 类的主动性
}class Gun implements Weapon {public void attack() { System.out.println("开枪"); } // 类的主动性
}
无论是 Sword 还是 Gun,它们的 attack() 逻辑由类自身定义,接口无法干预。
当通过接口调用方法时,本质是其他代码(如调用方)通过接口的抽象层间接使用类,而非接口本身在操作类。例如:
// 一个外部系统通过接口调用类
class GameEngine {void useWeapon(Weapon weapon) {weapon.attack(); // 实际调用 Sword.attack() 或 Gun.attack()}
}
这里
GameEngine是主动调用者,接口只是传递调用的媒介。
2. 生活类比修正
假设你是一个餐厅老板,你定义了一份“厨师岗位职责”(接口):职责要求:会做菜、会清洁厨房。
而具体的厨师 (类):张三、李四各自以自己的方式实现这些职责。
当顾客点菜时,他们通过“厨师岗位”这个抽象概念获得服务,但实际做菜的是张三或李四。岗位职责(接口)没有能力做菜,它只是确保张三李四具备做菜能力。
四、总结
你的困惑揭示了编程中“控制反转”的微妙之处:
直觉认为:接口是“控制者”(因为它规定了类必须做什么)。
实际逻辑:接口是“被控制者”(类通过实现接口,主动向外界暴露能力)。
这种反直觉的设计恰恰是面向对象编程的威力所在——通过抽象层解耦调用方和实现方,让系统更灵活。理解这一点后,你会意识到:不是接口在使用类,而是类通过接口向 外界 宣告“我能做什么”。
Part2. 编程接口与物理接口的区别
关于接口的知识,我们可能还有一些疑惑之处:
在java中,类如果实现了接口,那么接口类型的变量就可以操控对象的函数执行,感觉像是接口和对象在互动,一共有两个角色;而物理上的接口,比如可以用鼠标操控电脑,这就感觉是对象1与对象2在互动,它们之间的USB接口是互动的桥梁,这之中一共有三个角色。
为什么我们看到的只有接口和对象在互动?编程接口与物理接口的区别又是什么?
一、两者模式?三者模式!
接口作为规范,约束了对象必须提供哪些方法,但不参与实际交互过程。动态绑定机制隐藏了接口的被动性,让你感觉是接口和对象在“互动”的两者模式,但实际是对象在主动向外界提供服务。
在编程中,完整的交互确实涉及三个角色,它们分别是对象、接口、外界。其中外界既可以是我们这种调用者,也可以是其他调用该方法的代码。
以刚刚“生活类比修正”的比喻为例,编程接口就相当于一张岗位招聘,类就相当于具体的厨师,食客就是此处的外界。厨师按照招聘要求前来工作,做好菜品给食客品尝也展示了自己的实力。
三者的交互逻辑:(编程接口)
-
对象通过接口向外界宣告能力
类实现接口时,相当于对象对外界宣告:“我能提供这些服务!”(如Bird实现Flyable接口后,宣告自己能飞)。
主动性:对象的实现是服务提供的基础。 -
外界通过接口使用对象
调用方(外界)无需知道对象的具体类型,只需依赖接口定义的方法(如void makeItFly(Flyable flyableObj))。
间接性:接口作为“中间人”确保调用的合法性,但不参与实际执行。 -
接口作为契约约束双方
-
对对象:必须实现接口方法。
-
对外界:只能调用接口定义的方法。
-
交互流程图:
外界(调用方) —— 通过接口 ——→ 对象↑ ↓←— — 实际执行结果 —— ←—
二、核心区别
我们再来看一下物理接口的三者模式:
-
参与者:设备A(如鼠标)、设备B(如电脑)、接口标准(如USB协议 + 物理接口)。
-
互动逻辑:
物理接口是独立存在的中间层,它同时包含:-
逻辑规范:定义数据传输格式(如USB协议规定电压、信号时序)。
-
物理实体:接口形状(如Type-C插头)、线缆、电路。
设备A和B必须共同遵守接口规范,并通过物理接口交换数据。此时接口既是“约束者”又是“传输通道”。
-
编程接口和物理接口都有约束的作用,为什么只有物理接口有信息传输的职责?(编程接口与物理接口的核心区别):
1. 抽象层次的分离
-
编程接口:属于纯逻辑层,仅通过代码约定行为,无需物理载体。
-
物理接口:需要逻辑与物理的统一,既要定义交互规则(如协议),又要实现物理信号传输。
2. 交互复杂性的需求
-
物理设备间的交互需要严格的兼容性(如电压匹配、插头尺寸),因此必须通过独立的接口层协调。
-
编程中的接口即便隐藏了外界角色的存在,也足够支撑多态性。
3. 设计目标的差异
-
编程接口:目标是解耦(调用方无需关心具体实现类)。
-
物理接口:目标是互联(让不同设备能够物理连接并通信)。
“用接口类型变量调用方法”的本质是:类通过接口向外界提供服务,而调用方通过接口的抽象层使用这些服务。接口是被动的约束者,类才是主动的提供者(调用者是外界)。 这种设计让代码更灵活,但抽象层带来的“间接性”会让人产生方向混淆——这是学习抽象思维的必经之路,而非认知障碍。
让我们在学习编程的道路上继续成长吧!本期分享完毕,感谢大家的支持Thanks♪(・ω・)ノ

相关文章:
假面与演员:到底是接口在使用类,还是类在使用接口?编程接口与物理接口的区别又是什么?
前言:本篇文章解释了接口学习过程中的2个常见问题,一个是“为什么是类在使用接口”,另一个一个是“编程接口与物理接口的差异源于所处的抽象层次和交互模式的不同”,旨在揭示编程接口的本质。 Part1.是类在使用接口 当学习接口时…...
Node.js 中的 Event 模块详解
Node.js 中的 Event 模块是实现事件驱动编程的核心模块。它基于观察者模式,允许对象(称为“事件发射器”)发布事件,而其他对象(称为“事件监听器”)可以订阅并响应这些事件。这种模式非常适合处理异步操作和…...
C# 添加图标
一、前言 为应用程序添加图标是优化用户界面、提升应用辨识度的重要操作。合适的图标能帮助用户快速识别和区分不同应用,增强应用的易用性和专业性。 本指南旨在为你提供详细、易懂的步骤,教你如何为应用程序的窗体添加图标。从图标素材的获取到具体的…...
Docker 入门与实战:从安装到容器管理的完整指南
🚀 Docker 入门与实战:从安装到容器管理的完整指南 🌟 📖 简介 在现代软件开发中,容器化技术已经成为不可或缺的一部分。而 Docker 作为容器化领域的领头羊,以其轻量级、高效和跨平台的特性,深…...
4.【线性代数】——矩阵的LU分解
四 矩阵的LU分解 1. AB的逆矩阵2. 转置矩阵3. ALU3.1 2x2矩阵3.2 3x3矩阵3.3 nxn的矩阵分解的次数? 1. AB的逆矩阵 { ( A B ) ( B − 1 A − 1 ) I ( B − 1 A − 1 ) ( A B ) I ⇒ ( A B ) − 1 B − 1 A − 1 \begin{cases} (AB)(B^{-1}A^{-1}) I\\ (B^{-1}A^…...
ELK8.17部署(Ubantu24x64)
检查java环境 ELK8.x不支持java8 若无环境可执行 sudo apt install openjdk-17-jre-headless 准备安装包 官网下载地址: ELK products 搜Elasticsearch、Kibana、Logstash、Filebeat versions需一致,这里使用8.17.0 Elasticsearch Kibana Logstash Filebeat e…...
什么是算法的空间复杂度和时间复杂度,分别怎么衡量。
1. 时间复杂度 时间复杂度衡量的是算法运行时间与输入规模之间的关系。它通常用大O记号(Big O Notation)表示,例如 O(1)、O(n)、O(n2) 等。 衡量方法: 常数时间复杂度 O(1):无论输入规模如何,算法的执行时…...
HCIA项目实践---ACL访问控制列表相关知识和配置过程
十 ACL访问控制列表 1 策略的概念 在网络连通之后, 把所有为了追求控制而实现的技术都叫策略 2 访问控制 在路由器流量流入或者流出的接口上,匹配流量,执行相应的动作。(流量流入或者流出的接口并不是一个固定的概念而是一个相对的…...
细说STM32F407单片机RTC入侵检测和时间戳的原理及使用方法
目录 一、入侵检测的功能 二、示例功能 三、项目设置 1、晶振、DEBUG、CodeGenerator、USART6、KEYLED 2、RTC (1)设置RTC的模式。 (2)General、Time、Date\Wake Up分组 (3)Tamper分组 1ÿ…...
STM32 CAN过滤器配置和应用方法介绍
目录 概述 一、CAN过滤器核心概念 二、过滤器配置步骤(以标准ID为例) 三、不同模式的配置示例 四、高级配置技巧 五、调试与问题排查 六、关键计算公式 总结 概述 在STM32微控制器中,CAN过滤器可以配置为标识符屏蔽模式和标识符列表模…...
搜狗浏览器卸载教程
需求背景 今天发现geek居然无法卸载搜狗浏览器,作为一个老司机,这是不允许的。如果你使用geek或者windows的卸载,或者直接在它的安装包的Uninstall.exe中卸载,他走到100%就一直不动了。那玩意是假的。 卸载教程 结束 -----华丽的…...
Go 模块管理工具 `go mod tidy` 和 `go.sum` 文件详解
Go 模块管理工具 go mod tidy 和 go.sum 文件详解 引言 Go 语言自引入模块(module)系统以来,极大地简化了依赖管理和版本控制。go mod tidy 和 go.sum 文件是 Go 模块系统中的两个重要组成部分,它们共同确保项目的依赖项是最新的…...
音视频入门基础:RTP专题(9)——FFmpeg接收RTP流的原理和内部实现
一、引言 由《音视频入门基础:RTP专题(2)——使用FFmpeg命令生成RTP流》可以知道,推流端通过下面FFmpeg命令可以将一个媒体文件转推RTP,生成RTP流: ffmpeg -re -stream_loop -1 -i input.mp4 -vcodec cop…...
STM32 串口转 虚拟串口---实现USB转串口功能
一,USART与UART 区别 USART(Universal Synchronous/Asynchronous Receiver/Transmitter)通用同步/异步串行接收/发送器 相较于UART:通用异步收发传输器(Universal Asynchronous Receiver/Transmitter)多了…...
【进程与线程】Linux 线程、同步以及互斥
每个用户进程有自己的地址空间。 线程是操作系统与多线程编程的基础知识。 系统为每个用户进程创建一个 task_struct 来描述该进程:该结构体中包含了一个指针指向该进程的虚拟地址空间映射表: 实际上 task_struct 和地址空间映射表一起用来表示一个进程…...
胶囊网络动态路由算法:突破CNN空间局限性的数学原理与工程实践
一、CNN的空间局限性痛点解析 传统CNN的瓶颈: 池化操作导致空间信息丢失(最大池化丢弃85%激活值)无法建模层次空间关系(旋转/平移等变换不敏感)局部感受野限制全局特征整合 示例对比: # CNN最大池化示例…...
当pcie设备变化时centos是否会修改网络设备的名称(AI回答)
当pcie设备变化时centos是否会修改网络设备的名称 在CentOS(以及其他基于Linux的操作系统)中,网络接口的命名通常遵循特定的规则,尤其是在使用PCIe设备(如网络适配器)时。网络接口的命名通常基于设备的物理…...
【人工智能】释放数据潜能:使用Featuretools进行自动化特征工程
《Python OpenCV从菜鸟到高手》带你进入图像处理与计算机视觉的大门! 解锁Python编程的无限可能:《奇妙的Python》带你漫游代码世界 特征工程是机器学习流程中至关重要的一步,它直接影响模型的性能。然而,手动特征工程既耗时又需要领域专业知识。Featuretools是一个强大的…...
docker批量pull/save/load/tag/push镜像shell脚本
目录 注意: 脚本内容 执行效果 注意: 以下脚本为shell脚本通过docker/nerdctl进行镜像独立打包镜像的相关操作脚本内仓库信息和镜像存取路径需自行更改需自行创建images.txt并填写值,并且与脚本位于同级目录下 [rootmaster01 sulibao]# l…...
对正则表达式说不!!!
可能大家都会和我一样,时常会遇到正则表达式,有时候会忘记某些字符而苦恼。今天就帮助大家克服它,虽然不多,但我认为掌握这些足够了,万变不离其宗,以不变应万变。 一、正则表达式内容分类 1. 字符类 [abc…...
Redis日志分析
主从同步尝试: 日志中多次出现“Master is currently unable to PSYNC but should be in the future: -NOMASTERLINK Can’t SYNC while not connected with my master”。这表明从服务器尝试与主服务器进行部分重同步(PSYNC),但由…...
【做一个微信小程序】校园地图页面实现
前言 上一个教程我们实现了小程序的一些的功能,有背景渐变色,发布功能有的呢,已支持图片上传功能,表情和投票功能开发中(请期待)。下面是一个更高级的微信小程序实现,包含以下功能:…...
Web后端 - Maven管理工具
一 Maven简单介绍 Maven是apache旗下的一个开源项目,是一款用于管理和构建java项目的工具。 Maven的作用 二 Maven 安装配置 依赖配置 依赖传递 依赖范围 生命周期 注意事项:在同一套生命周期中,当运行后面的阶段时,前面的阶段都…...
20250217-POMO笔记
文章目录 前言一、伪代码一:POMO Training二、伪代码二:POMO Inference三、POMO注意力模型3.1、自注意力机制3.2、AM模型 前言 以下主要讲解两个算法的伪代码以及注意力模型。 一、伪代码一:POMO Training POMO Training是POMO模型训练的伪…...
JavaEE-SpringBoot快速入门
文章目录 本节目标Maven什么是Maven创建一个Maven项目maven项目功能maven的依赖管理全球仓库, 私服, 本地服务器, 配置国内镜像 第一个SpringBoot项目创建项目运行SpringBoot程序 SpringBoot原理初步Web服务器 总结 本节目标 了解什么是maven, 配置国内源使用Springboot创建项…...
游戏引擎学习第107天
仓库:https://gitee.com/mrxiao_com/2d_game_2 回顾我们之前停留的位置 在这段内容中,讨论了如何处理游戏中的三维效果,特别是如何处理额外的“Z层”。由于游戏中的艺术资源是位图而不是3D模型,因此实现三维效果变得非常具有挑战性。虽然可…...
Sprinig源码解析
前言 Spring 框架是 Java 企业级开发的基石,其源码设计体现了模块化、扩展性和灵活性。以下从 IoC 容器、AOP 实现、核心模块和关键设计模式四个角度对 Spring 源码进行深度解析,帮助理解其底层机制。即使Spring会使用的人见得就能使用。 一、IoC 容器源…...
ComfyUI流程图生图原理详解
一、引言 ComfyUI 是一款功能强大的工具,在图像生成等领域有着广泛应用。本文补充一点ComfyUI 的安装与配置过程遇到的问题,并深入剖析图生图过程及相关参数,帮助读者快速入门并深入理解其原理。 二、ComfyUI 的安装与配置中遇到的问题 &a…...
使用右侧值现象来处理一个word导入登记表的需求
需求也简单,导word文件用户登记表,有各部门的十几个版本(为什么这么多?不知道)。这里说下谈下我的一些代码做法: 需求分析: 如果能解决java字段和各项填的值怎么配对的问题,那么就…...
《open3d pyqt》Alpha重建
《open3d pyqt》Alpha重建 一、效果展示二、qt设置2.1 主界面添加动作2.2 dialog 界面、布局如下:三、核心代码一、效果展示 二、qt设置 2.1 主界面添加动作 2.2 dialog 界面、布局如下: 并生成py文件,参考前述章节 三、核心代码 main.py文件增加 from Su...
