为什么我不能在 Go 中用一种类型的切片替换另一种类型?
来源:Golang技术栈
时间:2023-04-13 09:04:32 297浏览 收藏
哈喽!大家好,很高兴又见面了,我是golang学习网的一名作者,今天由我给大家带来一篇《为什么我不能在 Go 中用一种类型的切片替换另一种类型?》,本文主要会讲到golang等等知识点,希望大家一起学习进步,也欢迎大家关注、点赞、收藏、转发! 下面就一起来看看吧!
问题内容
我试图了解 Go 的类型转换规则。假设我们有这些接口:
type woofer interface { woof() } type runner interface { run() } type woofRunner interface { woofer runner }
为了满足接口,我们有一个dog
类型:
type dog struct{} func (*dog) run() {} func (*dog) woof() {}
这两个函数正在使用接口:
func allWoof(ws []woofer) {} func oneWoof(w woofer) {}
要使用这些方法,我可以编写以下内容:
dogs := make([]woofRunner, 10) oneWoof(dogs[0]) allWoof(dogs)
第一个功能oneWoof()
按预期工作;a*dog
实现了所有oneWoof
需求,它是一个woof
函数。
但是对于第二个函数allWoof
,Go 不会编译尝试的调用,报告如下:
不能在 allWoof 的参数中使用狗(类型 []woofRunner)作为类型 []woofer
使用类型转换也是不可能的;写作[]woofer(dogs)
也失败:
无法将狗(类型 []woofRunner)转换为类型 []woofer
的每个成员都[]woofRunner
具有满足 a 的所有必要功能[]woofer
,那么为什么禁止这种转换?
(我不确定这是否与 Go FAQ和
Stack Overflow
上的各种问题中解释的情况相同,在这些问题中人们询问将类型转换T
为interface{}
。切片/数组中的每个指针都指向一个可直接转换为的类型另一种类型。应该可以使用这些指针,原因与可以传递dog[0]
给“oneWoof”的原因相同。)
注意 1 :我知道一种解决方案是循环并逐个转换项目。我的问题是为什么这是必要的以及是否有更好的解决方案。
注2 :关于可分配性规则:
值 x 可分配给 T 类型的变量 [当] T 是接口类型并且 x 实现 T。
我们不能说如果切片/数组的类型可以分配给另一种类型,那么这些类型的数组也可以分配吗?
正确答案
除了 Go 拒绝沿此处其他答案中解决的这些方差关系转换切片之外,思考 为什么 Go 拒绝这样做是有用的,即使两种类型之间的内存表示相同。
在您的示例中,提供woofRunners
s 的切片作为类型参数[]woofer
要求对切片的元素类型进行 协变
处理。实际上,从切片中
读取 时,因为 awoofRunner
是 a woofer
,所以您知道 a 中存在的每个元素[]woofRunner
都会满足寻找
的读者[]woofer
。
然而,在 Go
中,切片是一种引用类型。当将切片作为参数传递给函数时,切片被复制,但在调用的函数体中使用的副本继续引用相同的后备数组(在append
超出其容量之前不需要重新分配)。数组的可变视图——通常,将一个项目插入到集合中——需要对元素类型进行
逆变 处理。也就是说,当需要一个函数参数以 插入 或 覆盖 类型的元素时woofRunner
,提供一个[]woofer
.
问题是函数是否要求切片参数
- 从中读取(对于读取
woofer
s,a[]woofRunner
与 a 一样好[]woofer
), - 写入它(对于写入
woofRunner
s,a 和 a[]woofer
一样好[]woofRunner
), - 或两者兼有(两者都不是另一个可接受的替代品)。
考虑一下如果 Go 确实以协变方式接受切片参数会发生什么,并且有人出现并进行了allWoof
如下更改:
// Another type satisfying `woofRunner`: type wolf struct{} func (*wolf) run() {} func (*wolf) woof() {} func allWoof(ws []woofer) { if len(ws) > 0 { ws[0] = &wolf{} } } dogs := []*dog{&dog{}, &dog{}} allWoof(dogs) // Doesn't compile, but what if it did?
即使 Go 愿意将 a[]*dog
视为 a []woofer
,我们也会*wolf
在我们的数组中得到 a
*dog
。一些语言通过对尝试的数组插入或覆盖进行运行时类型检查来防止这种意外,但由于 Go 甚至阻止我们做到这一点,它不需要这些额外的检查。
今天关于《为什么我不能在 Go 中用一种类型的切片替换另一种类型?》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于golang的内容请关注golang学习网公众号!
-
439 收藏
-
262 收藏
-
193 收藏
-
188 收藏
-
500 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
477 收藏
-
486 收藏
-
439 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 507次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习