为何在golang中,切片和映射之间的行为表现不一致?
来源:stackoverflow
时间:2024-03-20 11:36:37 354浏览 收藏
在 Go 语言中,切片和映射在行为上表现不一致。切片元素可寻址,而映射元素不可寻址。这是因为切片可以表示为数组的地址,而映射使用哈希算法存储键值对,其地址随着键值对的改变而改变。因此,支持映射元素的可寻址性会增加二进制大小并导致哈希算法频繁变化,从而影响性能。
type s struct { e int } func main() { a := []s{{1}} a[0].e = 2 b := map[int]s{0: {1}} b[0].e = 2 // error }
a[0] 可寻址,但 b[0] 不可寻址。
我知道第一个 0
是一个索引,第二个 0
是一个键。
为什么golang要这样实现?还有进一步考虑吗?
我已经在 github.com/golang/go/src/runtime 中阅读了 map
的源代码,如果 maxkeysize
和 maxvaluesize
足够小,则映射结构已经支持 indirectkey
和 indirectvalue
。
type maptype struct { ... keysize uint8 // size of key slot indirectkey bool // store ptr to key instead of key itself valuesize uint8 // size of value slot indirectvalue bool // store ptr to value instead of value itself ... }
我认为如果 golang 设计者想要这种语法,现在就很容易了。
当然indirectkey
indirectvalue
可能会花费更多的资源,gc也需要做更多的工作。
那么性能是支持这一点的唯一原因吗?
或者还有其他考虑吗?
在我看来,支持这样的语法很有价值。
解决方案
据我所知,
这是因为a[0]
可以替换为数组的地址。
同样,a[1]
可以替换为 a[0]+(keySize*1)
。
但是,如果映射不能这样做,哈希算法会根据您的键、值对及其数量而随时更改。
它们也会不时地重新排列。
需要进行具体的计算才能得到value的地址。
数组或切片很容易寻址,但对于映射来说,就像多个函数调用或结构查找......
如果你想用任何需要计算的东西来代替它,那么二进制大小将会增加几个数量级,而且哈希算法可能会不时发生变化。
好了,本文到此结束,带大家了解了《为何在golang中,切片和映射之间的行为表现不一致?》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多Golang知识!
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
477 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习