假面与演员:到底是接口在使用类,还是类在使用接口?编程接口与物理接口的区别又是什么?

前言:本篇文章解释了接口学习过程中的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接口是互动的桥梁,这之中一共有三个角色

为什么我们看到的只有接口和对象在互动?编程接口与物理接口的区别又是什么?

一、两者模式?三者模式!

接口作为规范,约束了对象必须提供哪些方法,但不参与实际交互过程。动态绑定机制隐藏了接口的被动性,让你感觉是接口和对象在“互动”的两者模式,但实际是对象在主动向外界提供服务

在编程中,完整的交互确实涉及三个角色,它们分别是对象、接口、外界其中外界既可以是我们这种调用者,也可以是其他调用该方法的代码

以刚刚“生活类比修正”的比喻为例,编程接口就相当于一张岗位招聘,类就相当于具体的厨师,食客就是此处的外界。厨师按照招聘要求前来工作,做好菜品给食客品尝也展示了自己的实力。

三者的交互逻辑:(编程接口)

  1. 对象通过接口向外界宣告能力
    类实现接口时,相当于对象对外界宣告:“我能提供这些服务!”(如 Bird 实现 Flyable 接口后,宣告自己能飞)。
    主动性:对象的实现是服务提供的基础。

  2. 外界通过接口使用对象
    调用方(外界)无需知道对象的具体类型,只需依赖接口定义的方法(如 void makeItFly(Flyable flyableObj))。
    间接性:接口作为“中间人”确保调用的合法性,但不参与实际执行

  3. 接口作为契约约束双方

    • 对对象:必须实现接口方法。

    • 对外界:只能调用接口定义的方法。

交互流程图:

外界(调用方) —— 通过接口 ——→ 对象↑                     ↓←— — 实际执行结果 ——  ←—

二、核心区别

我们再来看一下物理接口的三者模式:

  • 参与者:设备A(如鼠标)、设备B(如电脑)、接口标准(如USB协议 + 物理接口)。

  • 互动逻辑
    物理接口是独立存在的中间层,它同时包含:

    • 逻辑规范:定义数据传输格式(如USB协议规定电压、信号时序)。

    • 物理实体:接口形状(如Type-C插头)、线缆、电路。
      设备A和B必须共同遵守接口规范,并通过物理接口交换数据。此时接口既是“约束者”又是“传输通道”

编程接口和物理接口都有约束的作用,为什么只有物理接口有信息传输的职责?(编程接口与物理接口的核心区别):

1. 抽象层次的分离
  • 编程接口:属于纯逻辑层,仅通过代码约定行为,无需物理载体。

  • 物理接口:需要逻辑与物理的统一,既要定义交互规则(如协议),又要实现物理信号传输

2. 交互复杂性的需求
  • 物理设备间的交互需要严格的兼容性(如电压匹配、插头尺寸),因此必须通过独立的接口层协调。

  • 编程中的接口即便隐藏了外界角色的存在,也足够支撑多态性

3. 设计目标的差异
  • 编程接口:目标是解耦(调用方无需关心具体实现类)。

  • 物理接口:目标是互联(让不同设备能够物理连接并通信)。


“用接口类型变量调用方法”的本质是:类通过接口向外界提供服务,而调用方通过接口的抽象层使用这些服务。接口是被动的约束者,类才是主动的提供者(调用者是外界)。 这种设计让代码更灵活,但抽象层带来的“间接性”会让人产生方向混淆——这是学习抽象思维的必经之路,而非认知障碍。

让我们在学习编程的道路上继续成长吧!本期分享完毕,感谢大家的支持Thanks♪(・ω・)ノ

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

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

相关文章

python opencv基础使用总结

1.安装opencv库:pip install opencv-python 2.基础使用范例 import cv2 #图片的基本操作#1.读取一张图片 参数 1:图片的文件名如果图片放在当前文件夹下,直接写文件名就行,如lena.jpg否则需要给出绝对路径,如D:\Ope…

C++17 数学特殊函数:探索标准库中的强大工具

文章目录 1. 什么是数学特殊函数?2. C17 中的特殊函数2.1 贝塞尔函数2.2 勒让德函数2.3 椭圆积分2.4 伽马函数2.5 误差函数 3. 实际应用场景3.1 科学计算3.2 工程应用3.3 数据分析 4. 总结 在 C17 中,标准库引入了一系列数学特殊函数,这些函数…

mysql多主集群 galera cluster for mysql 8安装配置启动重启集群

[TOC] 一、安装mysql 1、安装 系统环境: Ubuntu 18.04 64位 MySQL 8.0.19 下载MySQL APT安装配置包 首先访问 https://dev.mysql.com/downloads/repo/apt/ 获取配置包下载地址 wget https://dev.mysql.com/get/mysql-apt-config_0.8.14-1_all.deb sudo dpkg -i mysq…

面向长范围交互式 LLM 智体的强化学习

25年2月来自 Apple 的论文“Reinforcement Learning for Long-Horizon Interactive LLM Agents”。 交互式数字智体 (IDA) 利用状态数字环境的 API 来执行任务以响应用户请求。虽然由指令调整的大语言模型 (LLM) 驱动 IDA 可以对多步交换中接口调用的反馈做出反应,…

sql语言语法的学习

sql通用语法 sql分类 DDL(操作数据库和表) 操作数据库 操作表_查询 操作表_创建 举例: 操作表_删除 操作表_修改 DML(增删改表中数据) DML添加数据 DML删除数据 DML修改数据 DQL 单表查询 基础查询 条件查询 案例演示: 排序查询 聚合函数 分组查询…

kkFileView二开之pdf转图片接口

kkFileView二开之Pdf转图片接口 kkFileView二开系列文章:1 kkFileView源码下载及编译2 Pdf转图片接口2.1 背景2.2 分析2.2 接口开发2.2.1 编写Pdf转图片方法2.2.2 编写转换接口 2.3 接口测试2.3.1 Pdf文件准备2.3.2 pdf2Image 3 部署 kkFileView二开系列文章&#x…

Electron 客户端心跳定时任务调度库调研文档 - Node.js 任务调度库技术调研文档

Electron 客户端心跳定时任务调度库调研文档 - Node.js 任务调度库技术调研文档 本文将对七个流行的定时任务调度库:node-cron、rxjs、bull、node-schedule、agenda、bree、cron。这些库都可以用来处理定时任务,但它们的特点和适用场景有所不同。我们将从…

驱动开发系列37 - Linux Graphics 2D 绘制流程(二)- 画布创建和窗口关联

一:概述 前面介绍Pixmap表示一块画布,是绘制发生的地方,本节看看驱动程序如何为画布分配内存/显存,以及如何与窗口关联的。 二:为画布分配BO 在系统启动时(用户登录系统之后,会重启Xorg),在 Xorg 服务器初始化时,要为屏幕创建根窗口的 Pixmap,并绑定到 GPU framebu…

Numpy简易入门

1.1 认识Numpy数组对象 1.1.1 导入NumPy工具包 #导包并起别名为np import numpy as np1.1.2 创建一个三行五列的数组 data np.arange(15).reshap(3, 5)打印结果&#xff1a; 1.1.3 查询data类型 print(type(data))打印结果&#xff1a; <class numpy.ndarray> nda…

【设计模式】【行为型模式】访问者模式(Visitor)

&#x1f44b;hi&#xff0c;我不是一名外包公司的员工&#xff0c;也不会偷吃茶水间的零食&#xff0c;我的梦想是能写高端CRUD &#x1f525; 2025本人正在沉淀中… 博客更新速度 &#x1f44d; 欢迎点赞、收藏、关注&#xff0c;跟上我的更新节奏 &#x1f3b5; 当你的天空突…

建筑兔零基础自学python记录22|实战人脸识别项目——视频人脸识别(下)11

这次我们继续解读代码&#xff0c;我们主要来看下面两个部分&#xff1b; 至于人脸识别成功的要点我们在最后总结~ 具体代码学习&#xff1a; #定义人脸名称 def name():#预学习照片存放位置path M:/python/workspace/PythonProject/face/imagePaths[os.path.join(path,f) f…

源代码防泄密沙箱是啥意思?

SDC沙盒通过多种技术手段实现环境隔离&#xff0c;从而有效防止数据泄露。以下是其具体的隔离机制&#xff1a; 1. 创建隔离的加密沙盒 SDC沙盒在员工电脑上虚拟出一个对外隔绝的加密沙盒。这个沙盒会主动与服务器进行认证对接&#xff0c;形成服务器-客户端沙盒这样一个涉密…

【复现DeepSeek-R1之Open R1实战】系列4:SFT和GRPO源码逐行深度解析(上)(3万字长文,从零开始到入门,包含详细的拓展基础知识)

目录 1 前言1.1 Open R1项目简介1.2 主要步骤1.3 原理图 2 基础知识2.1 Vocabulary和Tokenizer2.1.1 vocab.json, tokenizer.json, tokenizer_config.json2.1.2 什么是tokenizer2.1.3 在哪一步将tokenizer转成embedding2.1.4 tokenizer的代码实现 2.2 SFT和GRPO2.2.1 SFT2.2.2 …

课题推荐:高空长航无人机多源信息高精度融合导航技术研究

高空长航无人机多源信息高精度融合导航技术的研究&#xff0c;具有重要的理论意义与应用价值。通过深入研究多源信息融合技术&#xff0c;可以有效提升无人机在高空复杂环境下的导航能力&#xff0c;为无人机的广泛应用提供强有力的技术支持。希望该课题能够得到重视和支持&…

python_excel批量插入图片

提取excel的指定列的值的后4位&#xff08;数值&#xff09;&#xff0c;在其它列名的单元格中&#xff0c;批量嵌入与该数值匹配的图片&#xff08;未实现居中&#xff09;&#xff0c;每间隔4行处理一次&#xff08;合并过单元格&#xff09;。 import pandas as pd from ope…

DeepSeek 助力 Vue 开发:打造丝滑的颜色选择器(Color Picker)

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享一篇文章&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495; 目录 Deep…

Jenkinsdebug:遇到ERROR: unable to select packages:怎么处理

报错信息&#xff1a; 报错信息解释&#xff1a; musl-1.2.5-r0 和 musl-dev-1.2.5-r1: 这里说明 musl-dev 需要一个特定版本的 musl&#xff0c;即 musl1.2.5-r1&#xff0c;但是当前版本的 musl&#xff08;1.2.5-r0&#xff09;并不满足这个条件。版本冲突: 当尝试安装新…

并查集基础+优化(下标从0开始)

#include<iostream> #include<algorithm> #include<vector> using namespace std; const int N 1e510; int n,m; int fa[N]; void set(int u,int v) {fa[v] u; } int find(int arr[],int i) {while(arr[i] ! -1){i arr[i]; } return i;//返回的是这个节点…

STM32创建静态库lib

创建静态库lib 1. 新建工程1.1 创建工程文件夹1.2 编写用户相关代码1.2.1 stm32f4xx_it.h1.2.2 stm32f4xx_it.c1.2.3 标准库配置&#xff1a;stm32f4xx_conf.h1.2.4 HAL库的配置&#xff1a;stm32f4xx_hal_conf.h1.2.5 LL库配置&#xff1a;stm32f4xx_ll_conf.h 1.3 移植通用文…

CV -- 基于GPU版显卡CUDA环境+Pycharm YOLOv8 检测

目录 下载 CUDA 下载 cuDNN 下载 anaconda 安装 PyTorch pycharm 搭配 yolo 环境并运行 阅读本文须知&#xff0c;需要电脑中有 Nvidia 显卡 下载 CUDA 打开 cmd &#xff0c;输入 nvidia-smi &#xff0c;查看电脑支持 CUDA 版本&#xff1a; 我这里是12.0&#xff0c;进入…