为何生成器执行速度低于回调函数?
来源:stackoverflow
时间:2024-03-27 13:00:33 286浏览 收藏
在 Go 语言中,使用生成器和回调函数迭代一系列值时,性能表现出显著差异。虽然二者均调用匿名函数相同次数,但生成器的执行速度明显低于回调函数,这一现象引起了困惑。经调查发现,性能差距与基准代码有关,使用匿名函数进行的延迟操作显著降低了生成器的性能。改进后的代码通过直接递增局部变量的方式取代延迟操作,获得了与回调函数更为接近的性能结果。
问题内容
我试图找出 go 中迭代一系列值最快的模式。每个都有明显的优点和缺点,但速度将是我的用例的一个重要因素
毫不奇怪,渠道是最慢的。
但是我很惊讶看到回调和生成器模式之间存在巨大差异,我试图了解发生了什么。在我看来,性能应该差不多,但事实并非如此。
两者都调用匿名函数的次数相同,但我想知道是否有与上下文切换相关的东西。
下层发生了什么?
我得到这些结果:
benchmarkiteratormethods/generator-8 31970 36196 ns/op benchmarkiteratormethods/callback-8 1000000 1193 ns/op benchmarkiteratormethods/channel-8 7999 148906 ns/op
我的基准代码:
package test
import (
"testing"
)
const iteratorTestIterations = 1000
func iteratorGeneratorFunctionForTests() func() (int, bool) {
i := 0
return func() (int, bool) {
for i < iteratorTestIterations {
defer (func() {
i++
})()
return i, false
}
return 0, true
}
}
func iteratorCallbackForTests(callback func(value int) bool) {
for i := 0; i < iteratorTestIterations; i++ {
shouldContinue := callback(i)
if !shouldContinue {
break
}
}
}
func iteratorChannelForTests() chan int {
channel := make(chan int)
go (func() {
for i := 0; i < iteratorTestIterations; i++ {
channel <- i
}
close(channel)
})()
return channel
}
func BenchmarkIteratorMethods(b *testing.B) {
b.Run("generator", func(b *testing.B) {
for i := 0; i < b.N; i++ {
iterator := iteratorGeneratorFunctionForTests()
for {
value, end := iterator()
if end {
break
}
_ = value
}
}
})
b.Run("callback", func(b *testing.B) {
for i := 0; i < b.N; i++ {
iteratorCallbackForTests(func(value int) bool {
_ = value
return true
})
}
})
b.Run("channel", func(b *testing.B) {
for i := 0; i < b.N; i++ {
for value := range iteratorChannelForTests() {
_ = value
}
}
})
}
解决方案
我找到了原因。这实际上与我的代码有关。使用函数延迟太慢。
我用这个替换了第一个函数:
func iteratorgeneratorfunctionfortests() func() (int, bool) {
i := 0
return func() (int, bool) {
for i < iteratortestiterations {
value := i
i++
return value, false
}
return 0, true
}
}
现在我得到了更合乎逻辑的结果:
BenchmarkIteratorMethods/generator-8 773698 1442 ns/op BenchmarkIteratorMethods/callback-8 966046 1168 ns/op BenchmarkIteratorMethods/channel-8 7710 145243 ns/op
理论要掌握,实操不能落!以上关于《为何生成器执行速度低于回调函数?》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!
声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
最新阅读
更多>
-
139 收藏
-
204 收藏
-
325 收藏
-
478 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
课程推荐
更多>
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习