Golang笔记——Interface类型

大家好,这里是Good Note,关注 公主号:Goodnote,专栏文章私信限时Free。本文详细介绍Golang的interface数据结构类型,包括基本实现和使用等。

在这里插入图片描述

文章目录

      • Go 语言中的 `interface` 详解
      • 接口定义
      • 实现接口
      • 空接口 `interface{}`
        • 示例:空接口的使用
      • interface 类型判断
      • 1. 类型断言(Type Assertion)
        • 语法
        • 示例
        • 类型转换和类型断言的区别
      • 2. 类型开关(Type Switch)
        • 语法
        • 示例
      • 3. 反射(Reflection)
        • 示例
        • 反射的关键点
      • 总结
      • 接口与多态
        • 解释
      • 接口的嵌套
        • 示例
      • 总结
    • 历史文章
      • MySQL数据库
      • Redis
      • Golang

Go 语言中的 interface 详解

Go 语言中的 interface 是一种强大且灵活的类型系统机制,它使得 Go 能够实现类似面向对象语言多态的特性。接口是一组方法签名的集合,而接口类型定义了某些行为,任何类型只要实现了接口中的方法,就自动成为该接口的实现类型。

简单的说,interface主要表现在以下几个方面:

  1. 方法的集合:接口是方法的集合,定义了一个类型需要实现哪些方法,但不关心实现的具体细节。只要一个类型实现了接口中的方法,就被认为实现了这个接口。
  2. 多态实现:接口的核心作用之一是实现多态。在 Go 中,不同类型只要实现了相同的接口方法,就可以通过接口来统一处理,达到灵活的多态性。这使得我们可以编写更加解耦和可扩展的代码。
  3. 空接口 interface{}是一种变量类型:接口是 Go 中的一个类型,空接口(interface{}) 是最基础的接口类型,它可以存储任何类型的值。实际上它是一个由两部分组成的结构体:
    • 类型(Dynamic Type):接口变量所保存值的具体类型。
    • 值(Dynamic Value):接口变量所保存的具体值。

在 Go 中,接口是非常核心的概念,它帮助我们编写解耦、灵活、可扩展的代码。

接口定义

接口定义了类型应该具备的行为(即方法)。Go 的接口与其他语言(如 Java 或 C++)中的接口有一些不同之处,特别是Go 的接口不需要显式声明实现,即只要类型实现了接口的方法,就自动实现了该接口。

type InterfaceName interface {Method1() // 方法签名Method2() // 方法签名
}

实现接口

Go 不要求显式声明某个类型实现了一个接口,只要该类型实现了接口中声明的所有方法,它就自然地“实现”了该接口。接口与类型之间的关系是隐式的。

package mainimport "fmt"// 定义接口
type Speaker interface {Speak() // 定义接口中的方法
}// 定义结构体
type Person struct {Name string
}// Person 实现了 Speaker 接口的 Speak 方法
func (p Person) Speak() {fmt.Println("Hello, my name is", p.Name)
}func main() {// 创建 Person 类型的实例p := Person{Name: "Alice"}// 将 p 赋给接口类型 Speakervar speaker Speaker = p// 调用接口的方法speaker.Speak()  // 输出:Hello, my name is Alice
}

说明:

  • Speaker 接口有一个方法 Speak
  • Person 类型实现了 Speak 方法,因此它自动实现了 Speaker 接口。
  • main 函数中,p 被赋给了接口类型 Speaker,然后调用接口的方法 Speak

空接口 interface{}

空接口 interface{} 是一个特殊的接口类型,它没有定义任何方法。由于任何类型都实现了空接口。空接口通常用于存储任何类型的值,类似于其他语言中的 Object 类型。

示例:空接口的使用
package mainimport "fmt"func main() {var x interface{} // 声明一个空接口x = 42            // x 可以存储 intfmt.Println(x)    // 输出:42x = "Hello"    // x 可以存储 stringfmt.Println(x) // 输出:Hellox = 3.14       // x 可以存储 float64fmt.Println(x) // 输出:3.14
}

interface 类型判断

在 Go 中,由于接口类型是通用的,它可以存储任何实现了该接口的类型,因此在使用接口时,可能并不知道它具体存储的是哪个类型的值。为了处理这种不确定性,Go 提供了三种常用的机制来检测或转换接口的实际类型:

  1. 类型断言(Type Assertion)
  2. 类型开关(Type Switch)
  3. 反射(Reflection)

这三者在 Go 中各有不同的用途,适用于不同的场景。通常情况下,如果你只需要判断一个接口的类型并进行相应处理,类型断言类型开关 是更常见的选择;而 反射 则通常用于更加动态和通用的场景,例如实现框架、库、ORM 等。

下面分别详细介绍这三种机制:

1. 类型断言(Type Assertion)

类型断言用于从接口类型转换回具体类型。它允许我们在运行时检查接口值的动态类型,并进行转换。

语法
value, ok := x.(T)
  • x 是接口类型的变量,T 是你想要转换成的具体类型。
  • 如果 x 存储的值是 T 类型,value 将会是存储的值,而 oktrue
  • 如果 x 存储的值不是 T 类型,value 会是 T 类型的零值,而 okfalse
示例
package mainimport "fmt"func main() {var a interface{} = 42// 类型断言if v, ok := a.(int); ok {fmt.Println("a is an int:", v)} else {fmt.Println("a is not an int")}
}

输出:

a is an int: 42
类型转换和类型断言的区别
  • 类型转换 是编译时的操作,用于在兼容类型之间进行显式转换。它适用于基本类型之间以及自定义类型的转换。
  • 类型断言 是运行时的操作,用于从接口类型中提取具体的类型和值。它适用于动态类型判断的场景。
属性类型转换 (Type Conversion)类型断言 (Type Assertion)
使用场景在兼容的类型之间进行显式转换(如 intfloat64)。用于从接口类型中提取底层的具体类型和值。
操作时机编译时。运行时。
适用范围基本数据类型、自定义类型等。接口类型(interface{} 或其他接口)。
结果将值转换为目标类型。提取接口的具体值或判断类型是否匹配。
错误处理不兼容类型转换会导致编译错误。不安全断言会导致运行时 panic;安全断言返回一个布尔值。
语法value := T(originalValue)value, ok := iface.(T)value := iface.(T)

2. 类型开关(Type Switch)

类型开关是 Go 中提供的更强大、更灵活的机制,它允许我们对接口值的类型进行多分支判断。与普通的 switch 语句不同,类型开关的 case 中是基于接口的动态类型进行匹配的。

语法
switch v := x.(type) {
case T1:// x 是 T1 类型
case T2:// x 是 T2 类型
default:// x 是其他类型
}
  • x.(type) 会检查 x 接口的动态类型。
  • 你可以根据不同的类型执行不同的逻辑。
示例
package mainimport "fmt"func identifyType(x interface{}) {switch v := x.(type) {case int:fmt.Println("int:", v)case string:fmt.Println("string:", v)default:fmt.Println("unknown type")}
}func main() {identifyType(42)        // 输出:int: 42identifyType("hello")    // 输出:string: helloidentifyType(3.14)       // 输出:unknown type
}

3. 反射(Reflection)

Go 的 reflect 包提供了在运行时操作接口的功能,能够动态地获取接口的具体类型和方法。这是 Go 中非常强大的特性,可以在不知道类型的情况下执行一些操作,例如获取类型的名称、字段信息、调用方法等。

示例
package mainimport ("fmt""reflect"
)type Dog struct{}func (d Dog) Speak() {fmt.Println("Woof!")
}func main() {var a interface{} = Dog{}// 使用反射获取类型t := reflect.TypeOf(a)fmt.Println("Type:", t)// 使用反射获取值v := reflect.ValueOf(a)fmt.Println("Value:", v)// 通过反射调用方法if t.Kind() == reflect.Struct {method := v.MethodByName("Speak")if method.IsValid() {method.Call(nil)  // 输出:Woof!}}
}
反射的关键点
  • reflect.TypeOf() 用来获取接口的具体类型。
  • reflect.ValueOf() 用来获取接口的具体值。
  • reflect.ValueOf(a).MethodByName("MethodName") 可以动态调用结构体的方法

总结

  • 类型断言:用于在运行时提取接口的具体类型值,如果类型不匹配,可以使用 ok 变量避免运行时错误。
  • 类型开关:允许你对接口值的动态类型进行多分支判断,可以在多个可能的类型之间选择。
  • 反射:通过 reflect 包可以在运行时获取接口的类型和值,甚至可以动态地调用方法或修改值

接口与多态

Go 语言的多态是通过接口实现的。接口提供了一种方法,让不同类型的对象能以统一的方式来调用它们的行为。

package mainimport "fmt"type Animals interface {Say()
}type Dog struct{}
type Cat struct{}func (d Dog) Say() {fmt.Println("wangwang")
}func (c Cat) Say() {fmt.Println("miaomiao")
}func main() {var d Dogd.Say() // 输出:wangwangvar c Catc.Say() // 输出:miaomiao// 使用接口变量1,可以接受任何实现了 Say() 方法的类型var a Animalsa = da.Say() // 输出:wangwanga = ca.Say() // 输出:miaomiao// 使用接口变量2,可以接受任何实现了 Say() 方法的类型var a1 Animalsa1 = Dog{}a1.Say() // 输出:wangwanga1 = Cat{}a1.Say() // 输出:miaomiao
}
解释
  • DogCat 都实现了 Animals 接口。
  • a 是一个接口类型,可以存储任何实现了 Say 方法的类型。
  • 通过多态,我们可以使用同一个接口变量 a 存储不同的类型,并调用它们各自的 Say 方法。

接口的嵌套

Go 允许接口嵌套,接口可以继承其他接口的方法。当一个接口嵌套另一个接口时,它自动包含了被嵌套接口的方法。

示例
package mainimport "fmt"// 定义 Animal 接口
type Animal interface {Speak()
}// 定义 Worker 接口,嵌入 Animal 接口
type Worker interface {Animal  // Animal 接口被嵌套在 Worker 接口中Work()
}// 定义 Dog 结构体
type Dog struct{}// Dog 实现了 Animal 接口的 Speak 方法
func (d Dog) Speak() {fmt.Println("wangwang")
}// Dog 实现了 Worker 接口的 Work 方法
func (d Dog) Work() {fmt.Println("Dog is working!")
}func main() {// 创建 Dog 类型的对象var w Worker = Dog{}// 调用 Worker 接口的方法w.Speak() // 输出:wangwangw.Work()  // 输出:Dog is working!
}
  1. 接口嵌套Worker 接口通过 Animal 接口嵌套了 Speak 方法,这意味着 Worker 接口需要实现 SpeakWork 方法。

  2. Dog 类型实现接口Dog 类型实现了 SpeakWork 方法,满足了 Worker 接口的要求。

  3. 接口引用:在 main 函数中,wWorker 类型的接口变量,它引用了 Dog 类型的对象。由于 Dog 类型实现了 SpeakWork 方法,所以可以调用 w.Speak()w.Work()

总结

  • Go 中的接口:接口是一组方法签名的集合,Go 语言通过接口实现了多态。
  • 隐式实现:Go 中不需要显式声明类型实现接口,任何实现了接口方法的类型都会自动实现该接口。
  • 空接口interface{} 可以存储任何类型的值,类似于其他语言中的 Object 类型。
  • 类型断言:允许从接口类型转换回具体类型,提供灵活的运行时类型检查。
  • 接口与多态:通过接口,Go 实现了动态多态,允许不同类型通过统一的接口调用各自的行为。

接口是 Go 语言的核心特性之一,它使得 Go 在保持简洁和灵活性的同时,支持面向对象的编程风格。

历史文章

MySQL数据库

MySQL数据库

Redis

Redis数据库笔记合集

Golang

  1. Golang笔记——语言基础知识
  2. Golang笔记——切片与数组
  3. Golang笔记——hashmap
  4. Golang笔记——rune和byte
  5. Golang笔记——channel

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

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

相关文章

轨迹优化 | 基于贝塞尔曲线的无约束路径平滑与粗轨迹生成(附ROS C++/Python仿真)

目录 0 专栏介绍1 从路径到轨迹2 基于贝塞尔曲线的粗轨迹生成2.1 路径关键点提取2.2 路径点航向角计算2.3 贝塞尔曲线轨迹生成 3 算法仿真3.1 ROS C仿真3.2 Python仿真 0 专栏介绍 🔥课设、毕设、创新竞赛必备!🔥本专栏涉及更高阶的运动规划…

理解STC15F2K60S2单片机的最小电路

一、STC15F2K60S2与51单片机的区别 STC15F2K60S2和51单片机虽然都基于8051内核,但在多个方面存在显著区别: 1. CPU性能: - STC15F2K60S2:采用增强型8051 CPU,1T单时钟/机器周期,速度比普通8051快8-12倍…

VSCode 搜索 搜不到

VSCode 搜索 搜不到 今天打开一个新的工作目录之后 ctrl P 搜文件 搜不到 经观察后发现 当我搜索时候, 右侧搜索按钮有一个时钟标识,疑似 搜索的范围 是最近打开内容。 经过和全局搜索的编辑器对比设置后发现,把设置中 下图中 选项去掉勾选…

软件测试 —— Selenium常用函数

软件测试 —— Selenium常用函数 操作测试对象点击/提交对象 click()模拟按键输入 send_keys("")清除文本内容 clear() 模拟用户键盘行为 Keys包示例用法 获取文本信息 textget_attribute("属性名称") 获取当前页面标题 title获取当前页面的 url current_u…

Vue 学习之旅:核心技术学习总结与实战案例分享(vue指令下+计算属性+侦听器)

Vue 学习之旅:核心技术学习总结与实战案例分享 文章目录 Vue 学习之旅:核心技术学习总结与实战案例分享一、指令补充(一)指令修饰符(二)v-bind 对样式操作的增强(三)v-model 应用于其…

UE5 打包项目

UE5 打包项目 flyfish 通过 “文件”->“打开项目”,然后在弹出的对话框中选择项目文件(通常是以.uproject为后缀的文件) 选择目标平台: 在 UE5 主界面中,找到 “平台”(Platforms)。根据…

1. Doris分布式环境搭建

一. 环境准备 本次测试集群采用3台机器hadoop1、hadoop2、hadoop3, Frontend和Backend部署在同一台机器上,Frontend部署3台组成高可用,Backend部署3个节点,组成3副本存储。 主机IP操作系统FrontendBackendhadoop1192.168.47.128Centos7Foll…

win10电脑 定时关机

win10电脑 定时关机 https://weibo.com/ttarticle/p/show?id2309405110707766296723 二、使用任务计划程序设置定时关机打开任务计划程序: 按下“Win S”组合键,打开搜索框。 在搜索框中输入“任务计划程序”,然后点击搜索结果中的“任务…

day07_Spark SQL

文章目录 day07_Spark SQL课程笔记一、今日课程内容二、Spark SQL函数定义(掌握)1、窗口函数2、自定义函数背景2.1 回顾函数分类标准:SQL最开始是_内置函数&自定义函数_两种 2.2 自定义函数背景 3、Spark原生自定义UDF函数3.1 自定义函数流程&#x…

Hadoop3.x 万字解析,从入门到剖析源码

💖 欢迎来到我的博客! 非常高兴能在这里与您相遇。在这里,您不仅能获得有趣的技术分享,还能感受到轻松愉快的氛围。无论您是编程新手,还是资深开发者,都能在这里找到属于您的知识宝藏,学习和成长…

Java 实现 Elasticsearch 查询当前索引全部数据

Java 实现 Elasticsearch 查询当前索引全部数据 需求背景通常情况Java 实现查询 Elasticsearch 全部数据写在最后 需求背景 通常情况下,Elasticsearch 为了提高查询效率,对于不指定分页查询条数的查询语句,默认会返回10条数据。那么这就会有…

Elasticsearch ES|QL 地理空间索引加入纽约犯罪地图

可以根据地理空间数据连接两个索引。在本教程中,我将向你展示如何通过混合邻里多边形和 GPS 犯罪事件坐标来创建纽约市的犯罪地图。 安装 如果你还没有安装好自己的 Elasticsearch 及 Kibana 的话,请参考如下的链接来进行安装。 如何在 Linux&#xff0…

C#学习笔记 --- 简单应用

1.operator 运算符重载:使自定义类可以当做操作数一样进行使用。规则自己定。 2.partial 分部类: 同名方法写在不同位置,可以当成一个类使用。 3.索引器:使自定义类可以像数组一样通过索引值 访问到对应的数据。 4.params 数…

【Flink】Flink内存管理

Flink内存整体结构图: JobManager内存管理 JVM 进程总内存(Total Process Memory)Flink总内存(Total Flink Memory):JVM进程总内存减去JVM Metaspace(元空间)和JVM Overhead(运行时开销)上图解释: JVM进程总内存为2G;JVM运行时开销(JVM Overh…

MYSQL8创建新用户报错:You have an error in your SQL syntax;check...

本文所用——MYSQL版本:8.0.25 baidu都是直接创建新用户并赋权,如下: GRANT ALL PRIVILEGES ON *.* TO 用户名% IDENTIFIED BY 密码 WITH GRANT OPTION;但是我用的MYSQL版本它就不行,会报错! 经查阅资料发现——MY…

力扣经典练习题之198.打家劫舍

今天继续给大家分享一道力扣的做题心得今天这道题目是198.打家劫舍,这是一道非常经典的问题,在动态规划中经常考到这类问题,题目如下: 题目链接:198.打家劫舍 1,题目分析 首先此题就是给我们了一个数组,代表可以偷的房屋中的对应的金额,我们是一个小偷,一次可以偷很多…

万物互联的背后:MCU嵌入式硬件的奇幻之旅

文章背景:嵌入式硬件是什么? 你可能听说过嵌入式硬件,却总觉得它像是实验室里神秘的玩意儿。其实,它就在我们身边——从你手上的智能手表到车里的倒车雷达,无一不是嵌入式硬件的“杰作”。想象一块小小的电路板&#x…

LabVIEW数据库管理系统

LabVIEW数据库管理系统(DBMS)是一种集成了数据库技术与数据采集、控制系统的解决方案。通过LabVIEW的强大图形化编程环境,结合数据库的高效数据存储与管理能力,开发人员可以实现高效的数据交互、存储、查询、更新和报告生成。LabV…

如何在 Linux、MacOS 以及 Windows 中打开控制面板

控制面板不仅仅是一系列图标和菜单的集合;它是通往优化个人计算体验的大门。通过它,用户可以轻松调整从外观到性能的各种参数,确保他们的电脑能够完美地适应自己的需求。无论是想要提升系统安全性、管理硬件设备,还是简单地改变桌…

python 轮廓 获取环形区域

目录 效果图: 代码: 效果图: 代码: import cv2 import numpy as np# 读取图像 image cv2.imread(rE:\project\jijia\tools_jijia\img_tools\ground_mask.jpg, cv2.IMREAD_GRAYSCALE) # 二值化图像 # 二值化图像 _, binary cv…