当前位置: 首页 > article >正文

【Go语言基础【19】】接口:灵活实现多态的核心机制

文章目录

  • 零、概述
  • 一、接口基础
    • 1、接口的基本概念
      • a. 接口定义
      • b. 类型实现接口(无需显式声明)
      • c. 接口变量(体现了多态)
    • 2、实现接口的方式
    • 3、接口组合
    • 4、接口的底层结构
  • 二、空接口与类型断言
    • 1. 空接口(`interface{}`):接口类型的变量
    • 2. 类型断言:推断底层类型
    • 3. 类型开关(Type Switch)
  • 三、接口的应用场景
    • 1. 多态
    • 2. 依赖注入

零、概述

Go语言的接口(Interface)是一种抽象类型,用于定义一组方法的签名(即方法名、参数和返回值),但不包含方法的实现。接口是Go语言实现多态的核心机制,允许不同类型通过实现相同接口来表现出统一的行为。

Go的接口设计遵循**“鸭子类型”(Duck Typing)**原则:“如果它走路像鸭子,叫声像鸭子,那么它就是鸭子”。这种隐式实现方式使代码更灵活、松耦合,同时保持类型安全。通过合理使用接口,可以构建出可扩展、易维护的Go程序。

接口的注意事项

  1. 接口嵌套循环 接口不能直接或间接嵌套自身,否则会导致编译错误。
  2. 性能开销 接口调用涉及动态分发,比直接调用方法略慢(通常可忽略不计)。
  3. 避免过度抽象 仅在必要时使用接口,避免为简单场景引入过多抽象层。

 

一、接口基础

1、接口的基本概念

a. 接口定义

type Animal interface {Speak() string  // 方法签名,无实现Move() string
}

b. 类型实现接口(无需显式声明)

若类型(如结构体)实现了接口中的所有方法,则该类型自动实现了此接口,无需显式声明。

type Dog struct{}
func (d Dog) Speak() string { return "Woof!" }
func (d Dog) Move() string  { return "Run" }type Cat struct{}
func (c Cat) Speak() string { return "Meow!" }
func (c Cat) Move() string  { return "Jump" }

c. 接口变量(体现了多态)

接口类型的变量可以存储任何实现了该接口的类型的值,体现了多态的思想。

package main// 定义接口
type Speaker interface {Speak() string
}// 结构体Dog实现了Speaker接口
type Dog struct{}func (d Dog) Speak() string { return "Woof!" }// 结构体Cat也实现了Speaker接口
type Cat struct{}func (c Cat) Speak() string { return "Meow!" }func main() {var s Speaker      // 声明一个接口类型的变量s = Dog{}          // 存储Dog类型的值(因为Dog实现了Speaker)println(s.Speak()) // 输出: "Woof!"s = Cat{}          // 存储Cat类型的值(因为Cat也实现了Speaker)println(s.Speak()) // 输出: "Meow!"
}

 

2、实现接口的方式

a. 隐式实现 :无需显式声明类型实现了某个接口,只需实现接口中的所有方法。

b. 方法集规则

  • 值接收者方法T*T类型均实现该接口。
  • 指针接收者方法:仅*T类型实现该接口(需显式取地址)。
   type Mover interface {Move()}type Car struct{}func (c Car) Move() {}  // 值接收者方法,Car和*Car均实现Movertype Bike struct{}func (b *Bike) Move() {}  // 指针接收者方法,仅*Bike实现Movervar m Moverm = Car{}      // 合法m = &Bike{}    // 必须显式取地址// m = Bike{}   // 错误:Bike未实现Mover

 

3、接口组合

接口可通过组合其他接口形成新接口。

type Reader interface {Read(p []byte) (n int, err error)
}type Writer interface {Write(p []byte) (n int, err error)
}// 组合Reader和Writer
type ReadWriter interface {ReaderWriter
}

 

4、接口的底层结构

a. 接口变量在底层由两个字段组成:1. 动态类型(Type):存储实际值的类型、2. 动态值(Data):存储实际值的副本或指针。
b. 对于包含方法的接口(如Animal),Go使用itab(接口表)来关联接口方法和实际类型的方法实现。

 

二、空接口与类型断言

1. 空接口(interface{}):接口类型的变量

空接口不包含任何方法,所有类型都实现了空接口,因此可存储任意类型的值。

var x interface{}
x = 42          // 存储int
x = "hello"     // 存储string

 

2. 类型断言:推断底层类型

在 Go 语言中,类型断言(Type Assertion)是一种用于从接口值(interface)中提取其底层实际类型值的机制。比如:当你有一个接口变量时,有时需要知道它底层实际存储的是什么类型,并提取该类型的值。这时就需要使用类型断言。

语法结构

value, ok := interfaceVar.(TargetType)- interfaceVar:接口类型的变量。
- TargetType:你想要断言的目标类型。
- value:提取出的 TargetType 类型的值。
- ok:布尔值,表示断言是否成功(安全断言时使用)。
	var x interface{} = 42 // x 存储了 int 类型的值// 安全断言if v, ok := x.(int); ok {fmt.Println("x is int:", v) // 输出: x is int: 42}// 非安全断言(类型匹配时)v := x.(string) // interface {} is int, not stringfmt.Println(v)  // 输出: 42

 

3. 类型开关(Type Switch)

批量判断接口值的实际类型。

switch v := x.(type) {
case int:fmt.Println("x is int")
case string:fmt.Println("x is string")
default:fmt.Println("unknown type")
}

 

三、接口的应用场景

1. 多态

通过接口实现不同类型的统一行为。

func PrintAnimal(a Animal) {fmt.Println(a.Speak(), a.Move())
}PrintAnimal(Dog{})   // 输出: "Woof! Run"
PrintAnimal(Cat{})   // 输出: "Meow! Jump"

 

2. 依赖注入

通过接口解耦组件间的依赖关系。

type Logger interface {Log(msg string)
}type FileLogger struct{}
func (f FileLogger) Log(msg string) { /* 实现日志写入文件 */ }func ProcessData(l Logger) {l.Log("Processing data...")
}ProcessData(FileLogger{})  // 注入文件日志实现

 

相关文章:

【Go语言基础【19】】接口:灵活实现多态的核心机制

文章目录 零、概述一、接口基础1、接口的基本概念a. 接口定义b. 类型实现接口(无需显式声明)c. 接口变量(体现了多态) 2、实现接口的方式3、接口组合4、接口的底层结构 二、空接口与类型断言1. 空接口(interface{}&…...

MySql读写分离部署(一主一从,双主双从,Mycat)

参考资料: 参考视频 参考博客 视频参考资料及安装包: https://pan.baidu.com/s/1xT_WokN_xlRv0h06b6F3yg 提取码: aag3 Mysql主从复制部署指南(一主一从) NotePad++编辑Linux服务器文档 Mysql高版本(8.0及以后)Linux安装 Mysql分库分表(基于Mycat)的基本部署 …...

Go基本语法——go语言中的四种变量定义方法

前言 在go语言中,定义一个变量有四种方式,本文单从语法的层面来介绍这几种方式 单变量定义方法 1.var 变量名 类型,不进行初始化 例如,定义一个变量a后为其赋值,并且打印其值,运行结果如下 //1.不进行…...

27.【新型数据架构】-数据共享架构

27.【新型数据架构】-数据共享架构:降低数据获取成本,实时数据访问,保持数据新鲜度,促进数据经济发展,打破数据孤岛,标准化数据交换,增强数据安全性,完整审计追踪,合规性保障 一、数据共享架构的本质:打破壁垒的“数字立交桥” 传统企业或组织间的数据往往呈现“烟囱…...

virtualbox 如何虚拟机ip固定

1、在网络管理里新建 2、配置网络 3、 进入linux系统,查看 查看 网卡是enp0s8, ifconfig 4、进入网卡配置文件 cd /etc/sysconfig/network-scripts如果没有enp0s8 ,则使用mv ifcfg-enp0s3 ifcfg-enp0s8命令 配置项如下 TYPEEthernet PROXY_METHODn…...

RKNN3588上部署 RTDETRV2

RT-DETR V2 是由百度研究团队在 2024年 提出的,是其广受好评的实时目标检测模型 RT-DETR 的重大升级版本。它继承了第一代 RT-DETR 利用 Transformer 架构实现端到端目标检测 和 卓越实时性能 的核心优势,并针对模型精度、训练效率和部署灵活性进行了全方…...

Python----循环神经网络(BiLSTM:双向长短时记忆网络)

一、LSTM 与 BiLSTM对比 1.1、LSTM LSTM(长短期记忆网络) 是一种改进的循环神经网络(RNN),专门解决传统RNN难以学习长期依赖的问题。它通过遗忘门、输入门和输出门来控制信息的流动,保留重要信息并丢弃无关…...

Elasticsearch 常用操作命令整合 (cURL 版本)

Elasticsearch 常用操作命令整合 (cURL 版本) 集群管理 查看集群健康状态 curl -X GET "localhost:9200/_cluster/health?pretty"查看节点信息 curl -X GET "localhost:9200/_cat/nodes?v"查看集群统计信息 curl -X GET "localhost:9200/_clus…...

Redis持久化策略:RDB与AOF详解

目录 1. RDB持久化工作原理触发机制优点缺点配置示例 2. AOF持久化工作原理同步策略重写机制优点缺点配置示例 3. RDB与AOF比较4. 混合持久化(Redis 4.0)5. 选择建议 Redis提供了两种主要的持久化机制来保证数据安全:RDB(Redis Database)和AOF(Append Only File)。本…...

Linux系统编程-DAY10(TCP操作)

一、网络模型 1、服务器/客户端模型 (1)C/S:client server (2)B/S:browser server (3)P2P:peer to peer 2、C/S与B/S区别 (1)客户端不同&#…...

基于eclipse进行Birt报表开发

Birt报表开发最终实现效果: 简洁版的Birt报表开发实现效果,仅供参考! 可动态获取采购单ID,来打印出报表! 下面开始Birt报表开发教程: 首先:汉化的eclipse及Birt值得拥有:至少感觉上…...

GPU虚拟化

引言 现有如下环境(注意相关配置:只有一个k8s节点,且该节点上只有一张GPU卡): // k8s版本 $ kubectl version Client Version: version.Info{Major:"1", Minor:"22", GitVersion:"v1.22.7&…...

LabVIEW工业级多任务实时测控系统

采用LabVIEW构建了一套适用于工业自动化领域的多任务实时测控系统。系统采用分布式架构,集成高精度数据采集、实时控制、网络通信及远程监控等功能,通过硬件与软件的深度协同,实现对工业现场多类型信号的精准测控,展现 LabVIEW 在…...

Python学习(7) ----- Python起源

🐍《Python 的诞生》:一段圣诞假期的奇妙冒险 📍时间:1989 年圣诞节 在荷兰阿姆斯特丹的一个寒冷冬夜,灯光昏黄、窗外飘着雪。一个程序员 Guido van Rossum 正窝在家里度假——没有会议、没有项目、没有 bug&#xf…...

Java中List的forEach用法详解

在 Java 中,List.forEach() 是 Java 8 引入的一种简洁的遍历集合元素的方法。它基于函数式编程思想,接受一个 Consumer 函数式接口作为参数,用于对集合中的每个元素执行操作。 基本语法 java 复制 下载 list.forEach(consumer); 使用示…...

LeetCode 1356.根据数字二进制下1的数目排序

题目&#xff1a; 给你一个整数数组 arr 。请你将数组中的元素按照其二进制表示中数字 1 的数目升序排序。 如果存在多个数字二进制中 1 的数目相同&#xff0c;则必须将它们按照数值大小升序排列。 请你返回排序后的数组。 提示&#xff1a; 1 < arr.length < 5000…...

破解HTTP无状态:基于Java的Session与Cookie协同工作指南

HTTP协议自身是属于“无状态”协议 无状态是指&#xff1a;默认情况下&#xff0c;HTTP协议的客户端和服务器之间的这次通信&#xff0c;和下次通信之间没有直接的关系 但在实际开发中&#xff0c;我们很多时候是需要知道请求之间的关联关系的 上述图中的令牌&#xff0c;通常就…...

JS 事件流机制详解:冒泡、捕获与完整事件流

JS 事件流机制详解&#xff1a;冒泡、捕获与完整事件流 文章目录 JS 事件流机制详解&#xff1a;冒泡、捕获与完整事件流一、DOM 事件流基本概念二、事件捕获 (Event Capturing)特点代码示例 三、事件冒泡 (Event Bubbling)特点代码示例 四、完整事件流示例HTML 结构JavaScript…...

MYSQL too many connection问题排查和修复

1.连接数据库 mysql -u root -p 1.1 查看mysql路径 如果没有配置mysql的环境变量&#xff0c;可以直接找mysql的安装目录 打开任务管理器-》服务-》Mysql(根据版本不同后面带有数字&#xff0c;找运行的那个) 打开服务->mysql->属性-》可执行文件的路径&#xff0c;…...

SpringCloudAlibaba和SpringBoot版本问题

SpringCloudAlibaba和SpringBoot版本问题 直接参考官方给出的版本说明&#xff0c;具体地址&#xff1a;https://github.com/alibaba/spring-cloud-alibaba/wiki/%E7%89%88%E6%9C%AC%E8%AF%B4%E6%98%8E Spring Cloud Alibaba VersionSentinel VersionNacos VersionRocketMQ Ver…...

算法专题七:分治

快排 1.颜色分类 题目链接:75. 颜色分类 - 力扣(LeetCode) class Solution {public void swap(int[] nums, int i, int j){int t = nums[i];nums[i] = nums[j];nums[j] = t;}public void sortColors(int[] nums) {int left=-1 ,i=0 ,right=nums.length;while(i<right){i…...

Vue中虚拟DOM的原理与作用

绪论 首先我们先了解&#xff0c;DOM&#xff08;Document Object Model&#xff0c;文档对象模型&#xff09; 是浏览器对 HTML/XML 文档的结构化表示&#xff0c;它将文档解析为一个由节点&#xff08;Node&#xff09;和对象组成的树形结构&#xff08;称为 DOM 树&#xf…...

前端十种排序算法解析

1. 冒泡排序 1.1 说明 冒泡排序为一种常用排序算法&#xff0c;执行过程为从数组的第一个位置开始&#xff0c;相邻的进行比较&#xff0c;将最大的数移动到数组的最后位置执行的时间复杂度与空间复杂度为 o(n^2) 1.2 执行过程 从数组的第一个位置开始&#xff0c;截止位置为 …...

使用 C/C++ 和 OpenCV 添加图片水印

使用 C/C 和 OpenCV 添加图片水印 &#x1f5bc;️ 在数字图像处理中&#xff0c;添加水印是一种常见的操作&#xff0c;可以用于版权保护、品牌宣传或信息标注。本文将介绍如何使用 C/C 和强大的计算机视觉库 OpenCV 来实现将自定义水印&#xff08;图片或文字&#xff09;添…...

Secs/Gem第十二讲(基于secs4net项目的ChatGpt介绍)

好&#xff0c;那我们进入最关键的一讲—— 第十二讲&#xff1a;完整事件通知流程全景图——CEID 触发到主机接收的全过程 关键词&#xff1a;CEID 事件上报、S6F11 报文、事件触发流程、数据驱动机制、Report Dispatch、主机解析流程 本讲目标 你将彻底理解&#xff1a; 设…...

FastAPI实战起步:从Python环境到你的第一个“Hello World”API接口

上一篇文章中介绍了有关FastAPI的优势&#xff0c;本篇文章我将手把手带你从零开始&#xff0c;搭建FastAPI的开发环境&#xff0c;并成功运行你的第一个“Hello World”API。在开始之前&#xff0c;请确保你的电脑已经安装了Python 3.7或更高版本&#xff0c;以及VS Code&…...

使用vue3+ts+input封装上传组件,上传文件显示文件图标

效果图&#xff1a; 代码 <template><div class"custom-file-upload"><div class"upload"><!-- 显示已选择的文件 --><div class"file-list"><div v-for"(item, index) in state.filsList" :key&q…...

iOS 抖音导航栏首页一键分两列功能的实现

要实现 iOS 抖音首页导航栏的“一键分两列”功能&#xff08;通常指将单列内容切换为双列瀑布流布局&#xff09;&#xff0c;需结合自定义导航栏控件与布局动态切换逻辑。以下是关键实现步骤和技术要点&#xff0c;基于 iOS 原生开发框架&#xff08;Swift/Objective-C&#x…...

零基础入门 C 语言基础知识(含面试题):结构体、联合体、枚举、链表、环形队列、指针全解析!

&#x1f31f; 零基础入门 C 语言基础知识&#xff08;含面试题&#xff09;&#xff1a;结构体、联合体、枚举、链表、环形队列、指针全解析&#xff01; C 语言是所有程序员通向“系统世界”的第一把钥匙。很多嵌入式开发、操作系统内核、网络通信、图形引擎&#xff0c;背后…...

【Linux】Ubuntu 创建应用图标的方式汇总,deb/appimage/通用方法

Ubuntu 创建应用图标的方式汇总&#xff0c;deb/appimage/通用方法 对于标准的 Ubuntu&#xff08;使用 GNOME 桌面&#xff09;&#xff0c;desktop 后缀的桌面图标文件主要保存在以下三个路径&#xff1a; 当前用户的桌面目录&#xff08;这是最常见的位置&#xff09;。所…...