我终于识破了这个 Go 编译器把戏

2024-01-09 18:38

本文主要是介绍我终于识破了这个 Go 编译器把戏,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在 Go 语言的日常编码工作中,有一个非常普遍但诡异的编译错误,曾让我十分困惑。这个问题我相信不少 Gopher 都遇到过,不妨来看一下。

背景回顾

我们定义一个带有 WriteGoCode() 方法的 Gopher 接口,同时定义了 person 结构体,它存在 WriteGoCode() 方法。

type Gopher interface {WriteGoCode()
}type person struct {name string
}func (p person) WriteGoCode() {fmt.Printf("I am %s, i am writing go code!\n", p.name)
}

在 Go 语言中,只要某对象拥有接口的所有方法,那该对象即实现了该接口。pperson 结构体的实例化对象, Coding() 函数的入参是 Gopher 接口, person 对象实现了 Gopher 接口,因此 p 入参成功被运行。

func Coding(g Gopher) {g.WriteGoCode()
}func main() {p := person{name: "小菜刀"}Coding(p)
}// output:
I am 小菜刀, i am writing go code!

此时,我们将 Coding() 函数的入参改为 []Gopher 类型,入参为 []person

func Coding(gs []Gopher) {for _, g := range gs {g.WriteGoCode()}
}func main() {p := []person{{name: "小菜刀1号"},{name: "小菜刀2号"},}Coding(p)
}

但是,这个时候,编译却不能通过!

./main.go:29:8: cannot use p (type []person) as type []Gopher in argument to Coding

明明 person 类型实现了 Gopher 接口,且当函数入参为 Gopher 类型时,能够顺利被执行,但参数变为 []Gopher 时,却过不了编译,这是为什么?

语法通用规则

这个问题在 stackoverflow 上被热议,详情见文末参考链接1。

在 Go 中,有一个通用规则,即语法不应隐藏复杂/昂贵的操作。转换一个 stringinterface{} 它的时间复杂度是 O(1),转换 []stringinterface{} 同样也是一个 O(1) 操作,因为它还是一个单一值的转换。

如果要将 []string 转换为 []interface{},它是 O(N) 操作。因为切片的每个元素都必须转换为 interface{},这违背了 Go 的语法原则。

这个回答,你们同意吗?

当然,此规则存在一个例外:转换字符串。在将 string 转换为 []byte[]rune 时,即使需要 O(n) 操作,但 Go 会允许执行。

InterfaceSlice 问题

Ian Lance Taylor(Go 核心开发者) 在 Go 官方仓库中也回答了这个问题,详情见文末参考链接2。他给出了这样做的两个主要原因。

  • 原因一:类型为 []interface{} 的变量不是 interface!它仅仅是一个元素类型恰好为 interface{} 的切片。

  • 原因二:[]interface{} 变量有特定大小的内存布局,在编译期可知。这与 []MyType 是不同的。

每个 interface{} (运行时通过 runtime.eface 表示)占两个字长(一个字代表所包含内容的类型 _type,另外一个字表示所包含的数据 data 或者指向它的指针 )

因此,类型为 []interface{} 的长度为 N 的变量,它是由 N*2 个字长的数据块支持。而这与类型为 []MyType 的长度为 N 的变量的数据块大小是不同的,因为后者的数据块是 N*sizeof(MyType) 字长。

数据块的不同,造成的结果是编译器无法快速地将 []MyType 类型的内容分配给 []interface{} 类型的内容。

同理,[]Gopher 变量也是特定大小的内存布局(运行时通过 runtime.iface 表示)。这同样不能快速地将 []MyType 类型的内容分配给 []Gopher 类型。

因此,Ian Lance Taylor 回答闭环了 Go 的语法通用规则:Go 语法不应隐藏复杂/昂贵的操作,编译器会拒绝它们。

代码解决方案

再次将文章开头的例子附上,如果我们需要 [] person 类型的 p 能够成功入参 Coding() 函数,应该如何做呢。

func Coding(gs []Gopher) {for _, g := range gs {g.WriteGoCode()}
}func main() {p := []person{{name: "小菜刀1号"},{name: "小菜刀2号"},}Coding(p)
}

代码方案如下,核心是需要一个 []Gopher 类型的转换变量。

func main() {p := []person{{name: "小菜刀1号"},{name: "小菜刀2号"},}var interfaceSlice []Gopher = make([]Gopher, len(p))for i, g := range p {interfaceSlice[i] = g}Coding(interfaceSlice)
}// output:
I am 小菜刀1号, i am writing go code!
I am 小菜刀2号, i am writing go code!

总结

由于 []MyType[]interface{} 的转换,是昂贵的操作,Go 编译器不会允许这种情况通过编译,故而将这种开销的责任传递给开发者。

Go 是一门编译速度很快的语言,得益于它语法设计中贯彻着 “simpler is better” 的理念,这可不是说说而已。

参考链接

【1. Type converting slices of interfaces】

https://stackoverflow.com/questions/12753805/type-converting-slices-of-interfaces/12754757#12754757

【2. InterfaceSlice】

https://github.com/golang/go/wiki/InterfaceSlice

往期推荐

机器铃砍菜刀

欢迎添加小菜刀微信

加入Golang分享群学习交流!

感谢你的点赞在看哦~

这篇关于我终于识破了这个 Go 编译器把戏的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/588055

相关文章

go动态限制并发数量的实现示例

《go动态限制并发数量的实现示例》本文主要介绍了Go并发控制方法,通过带缓冲通道和第三方库实现并发数量限制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录带有缓冲大小的通道使用第三方库其他控制并发的方法因为go从语言层面支持并发,所以面试百分百会问到

Go语言并发之通知退出机制的实现

《Go语言并发之通知退出机制的实现》本文主要介绍了Go语言并发之通知退出机制的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1、通知退出机制1.1 进程/main函数退出1.2 通过channel退出1.3 通过cont

Go语言编译环境设置教程

《Go语言编译环境设置教程》Go语言支持高并发(goroutine)、自动垃圾回收,编译为跨平台二进制文件,云原生兼容且社区活跃,开发便捷,内置测试与vet工具辅助检测错误,依赖模块化管理,提升开发效... 目录Go语言优势下载 Go  配置编译环境配置 GOPROXYIDE 设置(VS Code)一些基本

使用Go实现文件复制的完整流程

《使用Go实现文件复制的完整流程》本案例将实现一个实用的文件操作工具:将一个文件的内容完整复制到另一个文件中,这是文件处理中的常见任务,比如配置文件备份、日志迁移、用户上传文件转存等,文中通过代码示例... 目录案例说明涉及China编程知识点示例代码代码解析示例运行练习扩展小结案例说明我们将通过标准库 os

深入理解Go语言中二维切片的使用

《深入理解Go语言中二维切片的使用》本文深入讲解了Go语言中二维切片的概念与应用,用于表示矩阵、表格等二维数据结构,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧... 目录引言二维切片的基本概念定义创建二维切片二维切片的操作访问元素修改元素遍历二维切片二维切片的动态调整追加行动态

go中的时间处理过程

《go中的时间处理过程》:本文主要介绍go中的时间处理过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1 获取当前时间2 获取当前时间戳3 获取当前时间的字符串格式4 相互转化4.1 时间戳转时间字符串 (int64 > string)4.2 时间字符串转时间

Go语言中make和new的区别及说明

《Go语言中make和new的区别及说明》:本文主要介绍Go语言中make和new的区别及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1 概述2 new 函数2.1 功能2.2 语法2.3 初始化案例3 make 函数3.1 功能3.2 语法3.3 初始化

Go语言中nil判断的注意事项(最新推荐)

《Go语言中nil判断的注意事项(最新推荐)》本文给大家介绍Go语言中nil判断的注意事项,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1.接口变量的特殊行为2.nil的合法类型3.nil值的实用行为4.自定义类型与nil5.反射判断nil6.函数返回的

Go语言数据库编程GORM 的基本使用详解

《Go语言数据库编程GORM的基本使用详解》GORM是Go语言流行的ORM框架,封装database/sql,支持自动迁移、关联、事务等,提供CRUD、条件查询、钩子函数、日志等功能,简化数据库操作... 目录一、安装与初始化1. 安装 GORM 及数据库驱动2. 建立数据库连接二、定义模型结构体三、自动迁

Go语言代码格式化的技巧分享

《Go语言代码格式化的技巧分享》在Go语言的开发过程中,代码格式化是一个看似细微却至关重要的环节,良好的代码格式化不仅能提升代码的可读性,还能促进团队协作,减少因代码风格差异引发的问题,Go在代码格式... 目录一、Go 语言代码格式化的重要性二、Go 语言代码格式化工具:gofmt 与 go fmt(一)