当前位置:首页 > 文章列表 > 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基础的同学学习。
    543次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    514次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    499次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • SEO  AI Mermaid 流程图:自然语言生成,文本驱动可视化创作
    AI Mermaid流程图
    SEO AI Mermaid 流程图工具:基于 Mermaid 语法,AI 辅助,自然语言生成流程图,提升可视化创作效率,适用于开发者、产品经理、教育工作者。
    534次使用
  • 搜获客笔记生成器:小红书医美爆款内容AI创作神器
    搜获客【笔记生成器】
    搜获客笔记生成器,国内首个聚焦小红书医美垂类的AI文案工具。1500万爆款文案库,行业专属算法,助您高效创作合规、引流的医美笔记,提升运营效率,引爆小红书流量!
    531次使用
  • iTerms:一站式法律AI工作台,智能合同审查起草与法律问答专家
    iTerms
    iTerms是一款专业的一站式法律AI工作台,提供AI合同审查、AI合同起草及AI法律问答服务。通过智能问答、深度思考与联网检索,助您高效检索法律法规与司法判例,告别传统模板,实现合同一键起草与在线编辑,大幅提升法律事务处理效率。
    554次使用
  • TokenPony:AI大模型API聚合平台,一站式接入,高效稳定高性价比
    TokenPony
    TokenPony是讯盟科技旗下的AI大模型聚合API平台。通过统一接口接入DeepSeek、Kimi、Qwen等主流模型,支持1024K超长上下文,实现零配置、免部署、极速响应与高性价比的AI应用开发,助力专业用户轻松构建智能服务。
    612次使用
  • 迅捷AIPPT:AI智能PPT生成器,高效制作专业演示文稿
    迅捷AIPPT
    迅捷AIPPT是一款高效AI智能PPT生成软件,一键智能生成精美演示文稿。内置海量专业模板、多样风格,支持自定义大纲,助您轻松制作高质量PPT,大幅节省时间。
    521次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码