当前位置:首页 > 文章列表 > Golang > Go问答 > 为什么我不能在 Go 中用一种类型的切片替换另一种类型?

为什么我不能在 Go 中用一种类型的切片替换另一种类型?

来源:Golang技术栈 2023-04-13 09:04:32 0浏览 收藏

哈喽!大家好,很高兴又见面了,我是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 上的各种问题中解释的情况相同,在这些问题中人们询问将类型转换Tinterface{}。切片/数组中的每个指针都指向一个可直接转换为的类型另一种类型。应该可以使用这些指针,原因与可以传递dog[0]给“oneWoof”的原因相同。)

注意 1 :我知道一种解决方案是循环并逐个转换项目。我的问题是为什么这是必要的以及是否有更好的解决方案。

注2 :关于可分配性规则:

值 x 可分配给 T 类型的变量 [当] T 是接口类型并且 x 实现 T。

我们不能说如果切片/数组的类型可以分配给另一种类型,那么这些类型的数组也可以分配吗?

正确答案

除了 Go 拒绝沿此处其他答案中解决的这些方差关系转换切片之外,思考 为什么 Go 拒绝这样做是有用的,即使两种类型之间的内存表示相同。

在您的示例中,提供woofRunnerss 的切片作为类型参数[]woofer要求对切片的元素类型进行 协变 处理。实际上,从切片中 读取 时,因为 awoofRunner 是 a woofer,所以您知道 a 中存在的每个元素[]woofRunner都会满足寻找 的读者[]woofer

然而,在 Go 中,切片是一种引用类型。当将切片作为参数传递给函数时,切片被复制,但在调用的函数体中使用的副本继续引用相同的后备数组(在append超出其容量之前不需要重新分配)。数组的可变视图——通常,将一个项目插入到集合中——需要对元素类型进行 逆变 处理。也就是说,当需要一个函数参数以 插入覆盖 类型的元素时woofRunner,提供一个[]woofer.

问题是函数是否要求切片参数

  • 从中读取(对于读取woofers,a[]woofRunner与 a 一样好[]woofer),
  • 写入它(对于写入woofRunners,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学习网公众号!

版本声明
本文转载于:Golang技术栈 如有侵犯,请联系study_golang@163.com删除
压缩输出不同于 Go to Ruby 实现压缩输出不同于 Go to Ruby 实现
上一篇
压缩输出不同于 Go to Ruby 实现
将 json 解组为类型
下一篇
将 json 解组为类型
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    511次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    498次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • AI边界平台:智能对话、写作、画图,一站式解决方案
    边界AI平台
    探索AI边界平台,领先的智能AI对话、写作与画图生成工具。高效便捷,满足多样化需求。立即体验!
    417次使用
  • 讯飞AI大学堂免费AI认证证书:大模型工程师认证,提升您的职场竞争力
    免费AI认证证书
    科大讯飞AI大学堂推出免费大模型工程师认证,助力您掌握AI技能,提升职场竞争力。体系化学习,实战项目,权威认证,助您成为企业级大模型应用人才。
    424次使用
  • 茅茅虫AIGC检测:精准识别AI生成内容,保障学术诚信
    茅茅虫AIGC检测
    茅茅虫AIGC检测,湖南茅茅虫科技有限公司倾力打造,运用NLP技术精准识别AI生成文本,提供论文、专著等学术文本的AIGC检测服务。支持多种格式,生成可视化报告,保障您的学术诚信和内容质量。
    560次使用
  • 赛林匹克平台:科技赛事聚合,赋能AI、算力、量子计算创新
    赛林匹克平台(Challympics)
    探索赛林匹克平台Challympics,一个聚焦人工智能、算力算法、量子计算等前沿技术的赛事聚合平台。连接产学研用,助力科技创新与产业升级。
    662次使用
  • SEO  笔格AIPPT:AI智能PPT制作,免费生成,高效演示
    笔格AIPPT
    SEO 笔格AIPPT是135编辑器推出的AI智能PPT制作平台,依托DeepSeek大模型,实现智能大纲生成、一键PPT生成、AI文字优化、图像生成等功能。免费试用,提升PPT制作效率,适用于商务演示、教育培训等多种场景。
    569次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码