Go基础篇:接口

news/2024/5/15 12:03:37/文章来源:https://blog.csdn.net/weixin_46618592/article/details/130436778

目录

  • 前言✨
  • 一、什么是接口?
  • 二、空接口 interface{}
    • 1、eface的定义
    • 2、需要注意的问题
  • 三、非空接口
    • 1、iface的定义
    • 2、itab的定义
    • 3、itab缓存

前言✨

前段时间忙着春招面试,现在也算告一段落,找到一家比较心仪的公司实习,开始慢慢回归状态,这后面几章我会学习go1.19版本的语言特性或者机制:类型系统、接口、断言以及反射的内容,也算是补上之前没有深入底层的内容。

一、什么是接口?

Go语言中的接口(interface)是一种类型,它定义了一组方法的集合,但没有具体的实现。接口可以被任何类型实现,只要该类型实现了接口中定义的所有方法。这种设计方式使得Go语言具有很高的灵活性和可扩展性。

// 定义一个接口,包含一个String方法
type Stringer interface {String() string
}// 定义一个结构体,实现Stringer接口
type Person struct {Name stringAge  int
}func (p Person) String() string {return fmt.Sprintf("%s (%d)", p.Name, p.Age)
}func TestInterface(t *testing.T) {// 定义一个接口变量,只能存储实现了Stringer接口的值var str Stringer// str = 42 // 编译错误,int类型没有实现Stringer接口// str = "hello" // 编译错误,string类型没有实现Stringer接口str = Person{"Bob", 30}fmt.Println(str) // Bob (30)
}

在这里插入图片描述

Go语言接口的特点是:

  • 接口是一种抽象的类型,它只定义了一组方法,而不指定具体的实现。
  • 接口是隐式实现的,也就是说,任何类型只要实现了接口的所有方法,就可以被认为是该接口的实例,而不需要显式地声明。
  • 接口可以组合,也就是说,一个接口可以包含另一个接口的所有方法,一个接口可以由多个接口的方法组成,从而实现接口的多态。

Go语言接口的优点是:

  • 接口可以提高代码的复用性和可维护性,因为它可以将不同类型的对象抽象为统一的接口,从而降低了代码之间的耦合度。
  • 接口可以提高代码的灵活性和扩展性,因为它可以支持多种实现方式,从而增加了代码的可变性和可选择性。
  • 接口可以提高代码的测试性和可测性,因为它可以方便地使用模拟对象或桩对象来替代真实对象,从而简化了单元测试和集成测试。

二、空接口 interface{}

Go语言中的接口是一种类型,它定义了一组方法的集合,接口可以分为空接口和非空接口。空接口(interface{})是一种特殊的接口类型,它没有任何方法,因此可以表示任何类型的值。非空接口则是指至少有一个方法的接口类型。

Jordan Oreilli 对空接口的一个很好的定义:

接口是两件事物:它是一组方法,但它也是一种类型。
空接口 interface{} 类型是没有方法的接口。由于 Go 语言没有 implements 关键字,所有类型都至少实现零个方法,并且接口是隐式实现的,所有类型都满足空接口。

在这里插入图片描述

1、eface的定义

在 Go 的泛型未敲定前,空接口和断言广泛用于实现泛型,空接口的底层数据结构是 eface,它是一个结构体,包含两个字段:_type和data。

eface结构体定义如下:

type eface struct {_type *_typedata  unsafe.Pointer
}

其中,_type是一个指向类型信息的指针,它包含了类型的名称、大小、对齐方式等信息。(不了解的请看上一节篇《Go基础篇:类型系统》),data是一个指向实际值的指针,它可以存储任何类型的值。

举个简单的例子:

type MyString string// 输出val的值和类型,这里只是稍微用到了反射reflect的TypeOf方法获取类型,后面我会详细讲的
func Print(val interface{}) {fmt.Println(val, reflect.TypeOf(val))
}func TestPrint(t *testing.T) {name := "linzy1"Print(name)var Myname MyString = "linzy2"Print(Myname)
}

在这里插入图片描述

例子中在空接口被赋值string类型和自定义类型时的值虽然相同,但是类型却不同,且都不是空接口 interface{} 类型。这是为什么呢?

当我们使用空接口来存储一个值时,Go语言会将该值的类型信息和实际值分别存储在eface结构体的_type和data字段中。在需要使用该值时,Go语言会根据_type字段中的类型信息,将data字段中的值转换为相应的类型,并进行相应的操作。

在这里插入图片描述

具体来说,当一个空接口变量被赋值为string类型的值时,Go语言会将该值的类型描述信息存储在该变量的_type字段中。这个类型描述信息包含了string类型的名称、大小、对齐方式等信息,可以通过反射包中的TypeOf()函数获取。

需要注意的是,由于空接口可以存储任何类型的值,因此在使用空接口时需要进行类型断言或类型转换,以确保程序的正确性。

2、需要注意的问题

空接口可以存储任何类型的值,但是 []interface{} 空接口切片类型不一样,[]interface{} 可以存储任何类型的值,但是不能直接赋值切片或者切片转换。

举个简单的例子:

func PrintAll(vals []interface{}) {fmt.Println("vals Type is:", reflect.TypeOf(vals))for _, val := range vals {fmt.Println(val, reflect.TypeOf(val))}
}func TestPrint(t *testing.T) {names := []string{"stanley", "david", "oscar"}// ./interface_test.go:144:11: cannot use names (variable of type []string) as type []interface{} in argument to PrintAll// PrintAll(names)ns := make([]interface{}, len(names))for i, name := range names {ns[i] = name}PrintAll(ns)ns = []interface{}{"linzy", 32, 1.68}PrintAll(ns)
}

在这里插入图片描述

在Go语言中,[]interface{}和[]string是两种不同的类型,它们之间不能直接赋值或转换。虽然[]interface{}可以存储任何类型的值,包括string类型,但是它本身并不是string类型的切片,因此不能直接赋值给[]string类型的变量。

最核心的点在于 slicetype 中的 elem 类型描述不同,所以他们是两种不同的类型。

type slicetype struct {typ  _typeelem *_type
}

三、非空接口

非空接口是指具有实际类型的接口,想要赋值给非空接口类型必须要实现该接口的所有方法,也就是说,它不是空接口。

1、iface的定义

在Go语言中,非空接口的底层数据结构是iface,它是一个结构体,定义如下:

type iface struct {tab  *itabdata unsafe.Pointer
}

其中,tab是一个指向方法表的指针,data是一个指向实际对象的指针。

2、itab的定义

itab是一个结构体,它定义了接口类型和实现类型之间的关系,包括接口类型的方法集合、实现类型的方法集合和方法表等信息。itab的定义如下:

type itab struct {inter *interfacetype_type *_typehash  uint32_     [4]bytefun   [1]uintptr
}

其中,inter是一个指向接口类型的指针,_type是一个指向动态类型的指针,hash是实现类型的哈希值,fun是一个指向方法表的指针数组,指向的是动态类型实现接口方法的地址

interfacetype和_type都是类型对象,它们包含了类型的名称、大小、对齐方式、方法集合等信息。uintptr是一个无符号整数类型,它可以存储指针类型的值。

itab结构体的大小是可变的,因为它的最后一个字段fun是一个指针数组,它的长度取决于实现类型的方法集合的大小。当实现类型的方法集合发生变化时,Go语言会重新生成一个新的itab结构体,并将其添加到缓存中以便后续可以复用。

举个简单的例子:

func TestReadWriter(t *testing.T) {// 定义io.ReadWriter接口类型的变量rw,需要实现Read和Write两个方法var rw io.ReadWriterf, _ := os.Open("linzy.txt")defer f.Close()rw = f// os.Stat函数获取文件的大小Stat, _ := f.Stat()tmp := make([]byte, Stat.Size())rw.Read(tmp)fmt.Println(string(tmp))
}

rw 赋值后
赋值的变化过程:

  • 当你将f \*os.File类型变量赋值给rw变量时,Go语言会生成一个新的itab结构体。它的inter字段指向io.ReadWriter接口类型的描述信息,_type字段指向*os.File类型的描述信息,fun字段指向一个方法表,其中包含了io.ReadWriter接口类型的需要的所有方法,拷贝的是方法地址。
  • Go语言会使用一个全局的哈希表来缓存itab结构体,这个itab结构体会被缓存起来以便后续可以复用。
  • 当你调用rw.Read(tmp)方法时,Go语言会根据rw变量对应的itab指针找到对应的方法表,并调用其中的Read方法。

总结:
当你将一个实现了接口类型的值赋给接口类型的变量时,Go语言会根据实际对象的类型和接口类型的方法集合生成一个新的itab结构体,并将其赋值给接口类型变量对应的itab指针。
这个itab结构体包含了接口类型的方法集合、实现类型的方法集合和方法表等信息,通过这些信息可以实现接口方法的调用。

3、itab缓存

一个非空接口类型和一个动态类型就可以确定一个itab的内容,容易出现重复的itab结构体。Go为了避免重复生成itab结构体的开销,并且可以减少内存的使用,提高程序的性能。
itab可复用

Go语言会使用一个全局的哈希表来缓存itab结构体,这个哈希表的键是一个由<接口类型, 动态类型>组成的key,值是对应的*itab结构体的指针。当我们创建一个非空接口变量时,Go语言会先在缓存中查找对应的itab结构体,如果找到了就直接使用,否则就动态生成一个新的itab结构体,并将其添加到缓存中以便后续可以复用。

在这里插入图片描述
但是这里说的哈希表跟go的map哈希表不相同,itab设计的哈希表结构更简单:

type itabTableType struct {size    uintptr             // length of entries array. Always a power of 2.count   uintptr             // current number of filled entries.entries [itabInitSize]*itab // really [size] large
}

需要一个itab时,会首先去itabTable里查找,计算哈希值时会用到接口类型(itab.inter)和动态类型(itab._type)的类型:

func itabHashFunc(inter *interfacetype, typ *_type) uintptr {// compiler has provided some good hash codes for us.return uintptr(inter.typ.hash ^ typ.hash)
}

如果能查询到对应的itab指针,就直接拿来使用。若没有就要再创建,然后添加到itabTable中。

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

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

相关文章

yum和repo详细解析

目录 一、rpm、yum、repo 二、repo文件详细解析 三、常用命令 四、更改epel.repo为清华源 一、rpm、yum、repo RPM RPM(Red-hat Package Manager)&#xff0c;是一个由红帽最早开发出来的包管理器&#xff0c;目前已经是大多数Linux发行的默认包管理器。RPM管理的包都是以…

程序员一个月拿两万,得知卖猪肉可以赚五万,你是选择做程序员还是卖猪肉?

在知乎上看到这么个帖子&#xff0c;觉得挺有意思&#xff0c;大家一起瞧瞧&#xff1f; 对此&#xff0c;我也看到了许多犀利的回答哈 **A&#xff1a;**我反过来问你&#xff0c;如果一对夫妇卖猪肉一个月只能挣一万&#xff0c;听说一名程序员一个月拿五万&#xff0c;他们…

刷题day66:目标和

题意描述&#xff1a; 给你一个整数数组 nums 和一个整数 target 。 向数组中的每个整数前添加 或 - &#xff0c;然后串联起所有整数&#xff0c;可以构造一个 表达式 &#xff1a; 例如&#xff0c;nums [2, 1] &#xff0c;可以在 2 之前添加 &#xff0c;在 1 之前添…

【简介】限流

限流 为什么要限流限流算法单机限流计数器算法滑动窗口算法漏桶算法令牌桶算法 分布式限流配额算法 限流策略限流位置 为什么要限流 作为有追求的程序员&#xff0c;我们都希望自己的系统跑的飞快&#xff0c;但是速度再快&#xff0c;系统处理请求耗时也不可能为0&#xff0c…

[MYAQL / Mariadb] 数据库学习-管理表记录2:匹配条件

管理表记录-匹配条件 匹配条件基本条件查询逻辑匹配&#xff08;多个条件判断&#xff09; 高级条件范围匹配模糊查询正则表达式&#xff1a; regexp四则运算 操作查询结果&#xff08;对查找到的数据再做处理&#xff09;排序分组&#xff08;一样的显示一次&#xff09;&…

【网络】交换机基本原理与配置

目录 &#x1f341;交换机工作原理 &#x1f341;交换机接口的双工模式 &#x1f341;交换机命令行模式 &#x1f341;交换机常见命令 &#x1f9e7;帮助命令 &#x1f9e7;常用命令介绍 &#x1f341;交换机的基本配置 &#x1f9e7;配置接口的双工模式及速率 &#x1f990;博…

janus videoroom 对接freeswitch conference 篇1

janus videoroom 实时性非常好&#xff0c; freeswitch conference的功能也很多 &#xff0c;有没办法集成到一块呢 让很多sip 视频终端也能显示到videoroom 里面&#xff0c; 实现方式要不两种 1.改源码实现 &#xff08;本文忽略 难度高&#xff09; 2.找一个videoroom管…

05mysql---函数

目录 1:日期函数 2:字符函数 3:数值函数 4:流程函数 1:日期函数 select 函数(参数) 函数功能举例curdate()返回当前日期2023-05-17curtime()返回当前时间14:44:33now()返回当前日期和时间2023-05-17 14:44:33year(date)获取指定date的年份month(date)获取指定date的月份day…

Python学习26:个人所得税计算器

描述‪‬‪‬‪‬‪‬‪‬‮‬‪‬‫‬‪‬‪‬‪‬‪‬‪‬‮‬‪‬‭‬‪‬‪‬‪‬‪‬‪‬‮‬‫‬‮‬‪‬‪‬‪‬‪‬‪‬‮‬‭‬‫‬‪‬‪‬‪‬‪‬‪‬‮‬‫‬‪‬‪‬‪‬‪‬‪‬‪‬‮‬‭‬‫‬‪‬‪‬‪‬‪‬‪‬‮‬‫‬‪‬ 2018年10月1日以前&#xff…

特征选择与特征提取

目录 一、 特征选择1、特征2、特征选择3、扩展——特征选择算法(有兴趣和精力可了解)拓展--完全搜索:拓展--启发式搜索:拓展--随机搜索:拓展--遗传算法: 二、 特征提取三、特征提取主要方法——PCA(主成分分析)1、PCA算法是如何实现的&#xff1f;PCA--零均值化&#xff08;中心…

【Linux命令】mount / umount命令、查看文件的挂载情况(lsblk)

在Windows环境下&#xff0c;我们可以直接访问检测到的外部设备&#xff0c;如磁盘、U盘等&#xff1b;然而在Linux环境下&#xff0c;外部硬件设备如磁盘、SD卡等外部设备是无法直接访问的。因此就需要挂载。 参考链接&#xff1a;mount 详解 目录 1、什么是挂载&#xff1f;…

Packet Tracer – 配置中继

Packet Tracer – 配置中继 地址分配表 设备 接口 IP 地址 子网掩码 交换机端口 VLAN PC1 NIC 172.17.10.21 255.255.255.0 S2 F0/11 10 PC2 NIC 172.17.20.22 255.255.255.0 S2 F0/18 20 PC3 NIC 172.17.30.23 255.255.255.0 S2 F0/6 30 PC4 NIC 1…

系统分析师:六、企业信息化战略与实施

目录 一、信息与信息化概念 1.1 信息的概念 1.2 信息化的概念 二、信息系统 2.1 信息系统的概念 2.2 信息系统的类型 2.3 信息系统的生命周期 2.4 系统建模 2.5 信息系统战略规划 2.6 信息化开发方法 三、电子政务 四、企业资源计划(ERP) 五、系统集成 一、信息与信息…

第三十三章 使用Redux管理状态

Redux&#xff08;全称为Redux&#xff09;是一个基于状态管理的JavaScript库&#xff0c;它可以用来构建可重用的、可维护的代码。Redux主要用于处理复杂的应用程序中的状态管理&#xff0c;它能够自动地处理应用程序中的更改&#xff0c;并在需要时更新视图。 Redux使用一种被…

FreeRTOS_系统配置

目录 1. FreeRTOSConfig.h 文件 2. "INCLUDE_" 开始的宏 2.1 INCLUDE_xSemaphoreGetMutexHolder 2.2 INCLUDE_xTaskAbortDelay 2.3 INCLUDE_vTaskDelay 2.4 INCLUDE_vTaskDelayUntil 2.5 INCLUDE_vTaskDelete 2.6 INCLUDE_xTaskGetCurrentTaskHandle 2.7 IN…

Matlab论文插图绘制模板第92期—折线图(Plot)

之前有分享过Matlab折线图的绘制模板&#xff1a; 但随着技术力的提升&#xff0c;发现很多地方还有待改进&#xff0c;于是便有了本期内容。 先来看一下成品效果&#xff1a; 特别提示&#xff1a;本期内容『数据代码』已上传资源群中&#xff0c;加群的朋友请自行下载。有需…

5th-Generation Mobile Communication Technology(二)

目录 一、5G/NR 1、 快速参考&#xff08;Quick Reference&#xff09; 2、5G Success 3、5G Challenges 4、Qualcomm Videos 二、PHY and Protocol 1、Frame Structure 2、Numerology 3、Waveform 4、Frequency Band 5、BWP 6、Synchronization 7、Beam Management 8、CSI Fra…

FiftyOne 系列教程(2)使用FiftyOne读取数据集

1. 支持的数据集 1.1. 支持各种常见的数据集格式 docs.voxel51.com/user guide/dataset creation/datasets.html#supported import formats此外&#xff0c;zoo上面有什么数据集&#xff0c;这里就可以加载到对应的数据集Available Zoo Datasets — FiftyOne 0.20.1 document…

nacos服务端源码集群同步源码分析

nacos集群状态同步源码分析 ServerStatusReporter ServerStatusReporter 是 ServerListManager的内部类 通过Component注解被解析到spring容器中 再通过PostConstruct初始化执行init方法 上边代码启动了一个延时2秒的线程 private class ServerStatusReporter implements Run…

微软限制我们使用Windows系统了,怎么办?

正如中国工程院院士倪光南所说&#xff0c;操作系统的成功与否&#xff0c;关键在于生态系统&#xff0c;需要搭建起完整的产业链上各个主体共生的生态体系。 当前我国国产操作系统市场发展很快&#xff0c;相比技术和市场突破&#xff0c;真正需要解决的问题是如何把生态建好…