Go 内联优化让程序员爱不释手
来源:脚本之家
时间:2023-01-07 12:16:01 177浏览 收藏
IT行业相对于一般传统行业,发展更新速度更快,一旦停止了学习,很快就会被行业所淘汰。所以我们需要踏踏实实的不断学习,精进自己的技术,尤其是初学者。今天golang学习网给大家整理了《Go 内联优化让程序员爱不释手》,聊聊优化、内联,我们一起来看看吧!
前言:
这是一篇介绍 Go 编译器如何实现内联的文章,以及这种优化将如何影响你的 Go 代码。
什么是内联?
内联是将较小的函数合并到它们各自的调用者中的行为。其在不同的计算历史时期的做法不一样,如下:
- 早期:这种优化通常是由手工完成的。
- 现在:内联是在编译过程中自动进行的一类基本优化之一。
为什么内联很重要?
内联是很重要的,每一门语言都必然会有。
具体的原因如下:
- 它消除了函数调用本身的开销。
- 它允许编译器更有效地应用其他优化策略。
核心来讲,就是性能更好了。
函数调用的开销
基本知识
在任何语言中调用一个函数都是有代价的。将参数编入寄存器或堆栈(取决于ABI),并在返回时反转这一过程,这些都是开销。
调用一个函数需要将程序计数器从指令流中的一个点跳到另一个点,这可能会导致流水线停滞。一旦进入函数,通常需要一些前言来为函数的执行准备一个新的堆栈框架,在返回调用者之前,还需要一个类似的尾声来退掉这个框架。
Go 中的开销
在 Go 中,一个函数的调用需要额外的成本来支持动态堆栈的增长。在进入时,goroutine 可用的堆栈空间的数量与函数所需的数量进行比较。
如果可用的堆栈空间不足,序言就会跳转到运行时逻辑,通过将堆栈复制到一个新的、更大的位置来增加堆栈。
一旦这样做了,运行时就会跳回到原始函数的起点,再次进行堆栈检查,现在通过了,然后继续调用。通过这种方式,goroutines可以从一个小的堆栈分配开始,只有在需要时才会增加。
这种检查很便宜,只需要几条指令,而且由于goroutine的堆栈以几何级数增长,检查很少失败。因此,现代处理器中的分支预测单元可以通过假设堆栈检查总是成功来隐藏堆栈检查的成本。在处理器错误预测堆栈检查并不得不丢弃它在投机执行时所做的工作的情况下,与运行时增长goroutine堆栈所需的工作成本相比,管道停滞的成本相对较小。
Go 里的优化
虽然每个函数调用的通用组件和 Go 特定组件的开销被使用投机执行技术的现代处理器很好地优化了,但这些开销不能完全消除,因此每个函数调用都带有性能成本,超过了执行有用工作的时间。由于函数调用的开销是固定的,较小的函数相对于较大的函数要付出更大的代价,因为它们每次调用的有用工作往往较少。
因此,消除这些开销的解决方案必须是消除函数调用本身,Go 编译器在某些条件下通过用函数的内容替换对函数的调用来做到这一点。这被称为内联,因为它使函数的主体与它的调用者保持一致。
改善优化的机会
Cliff Click 博士将内联描述为现代编译器进行的优化,因为它是常量传播和死代码消除等优化的基础。
实际上,内联允许编译器看得更远,允许它在特定函数被调用的情况下,观察到可以进一步简化或完全消除的逻辑。
由于内联可以递归应用,优化决策不仅可以在每个单独的函数的上下文中做出,还可以应用于调用路径中的函数链。
进行内联优化
不允许内联
内联的效果可以通过这个小例子来证明:
package main import "testing" //go:noinline func max(a, b int) int { if a > b { return a } return b } var Result int func BenchmarkMax(b *testing.B) { var r int for i := 0; i运行这个基准可以得到以下结果:
% go test -bench=.
BenchmarkMax-4 530687617 2.24 ns/op从执行结果来看,
max(-1, i)
的成本大约是 2.24ns,感觉性能不错。允许内联
现在让我们去掉
//go:noinline pragma
的语句,再看看不允许内联的情况下,性能是否会改变。如下结果:
% go test -bench=.
BenchmarkMax-4 1000000000 0.514 ns/op两个结果对比一看,2.24ns 和 0.51ns。差距至少一倍以上,根据 benchstat 的建议,内联情况下,性能提高了 78%。
如下结果:
% benchstat {old,new}.txt
name old time/op new time/op delta
Max-4 2.21ns ± 1% 0.49ns ± 6% -77.96% (p=0.000 n=18+19)这些改进从何而来?
首先,取消函数调用和相关的前导动作是主要的改进贡献者。其将 max 函数的内容拉到它的调用者中,减少了处理器执行的指令数量,并消除了几个分支。
现在 max 函数的内容对编译器来说是可见的,当它优化 BenchmarkMax 时,它可以做一些额外的改进。
考虑到一旦 max 被内联,BenchmarkMax 的主体对编译器而言就会有所改变,与用户端看到的并不一样。
如下代码:
func BenchmarkMax(b *testing.B) { var r int for i := 0; i i { r = -1 } else { r = i } } Result = r }再次运行基准测试,我们看到我们手动内联的版本与编译器内联的版本表现一样好。
如下结果:
% benchstat {old,new}.txt
name old time/op new time/op delta
Max-4 2.21ns ± 1% 0.48ns ± 3% -78.14% (p=0.000 n=18+18)现在,编译器可以获得 max 内联到 BenchmarkMax 的结果,它可以应用以前不可能的优化方法。
例如:编译器注意到 i 被初始化为 0,并且只被递增,所以任何与 i 的比较都可以假定 i 永远不会是负数。因此,条件
-1 > i
将永远不会为真。在证明了
-1 > i
永远不会为真之后,编译器可以将代码简化为:func BenchmarkMax(b *testing.B) { var r int for i := 0; i并且由于该分支现在是一个常数,编译器可以消除无法到达的路径,只留下如下代码:
func BenchmarkMax(b *testing.B) { var r int for i := 0; i通过内联和它所释放的优化,编译器已经将表达式
r = max(-1, i)
简化为r = i
。这个例子非常不错,很好的体现了内联的优化过程和性能提升的缘由。
内联的限制
在这篇文章中,讨论了所谓的叶子内联:将调用栈底部的一个函数内联到其直接调用者中的行为。
内联是一个递归的过程,一旦一个函数被内联到它的调用者中,编译器就可能将产生的代码内联到它的调用者中,依此类推。
例如如下代码:
func BenchmarkMaxMaxMax(b *testing.B) { var r int for i := 0; i该运行速度将会和前面的例子一样快,因为编译器能够反复应用上面的优化,将代码减少到相同的
r = i
表达式。总结
这篇文章针对内联进行了基本的概念介绍和分析,并且通过 Go 的例子进行了一步步的剖析,让大家对真实案例有了一个更贴切的理解。
Go 编译器的优化总是无处不在的。
本篇关于《Go 内联优化让程序员爱不释手》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注golang学习网公众号!
-
483 收藏
-
268 收藏
-
130 收藏
-
309 收藏
-
154 收藏
-
233 收藏
-
322 收藏
-
181 收藏
-
316 收藏
-
244 收藏
-
300 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 507次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习
-
- 诚心的黑裤
- 好细啊,码起来,感谢老哥的这篇文章,我会继续支持!
- 2023-01-11 14:47:59
-
- 坚定的故事
- 很有用,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢作者大大分享文章内容!
- 2023-01-10 17:44:44
-
- 爱笑的哈密瓜,数据线
- 很棒,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢up主分享技术贴!
- 2023-01-08 02:43:26
-
- 快乐的星月
- 这篇文章太及时了,太细致了,很好,码起来,关注博主了!希望博主能多写Golang相关的文章。
- 2023-01-08 01:14:01
-
- 威武的飞鸟
- 很详细,收藏了,感谢老哥的这篇技术贴,我会继续支持!
- 2023-01-07 21:13:22
-
- 喜悦的煎饼
- 这篇文章真是及时雨啊,好细啊,很好,收藏了,关注up主了!希望up主能多写Golang相关的文章。
- 2023-01-07 16:44:31
-
- 还单身的康乃馨
- 这篇文章内容真是及时雨啊,博主加油!
- 2023-01-07 16:13:33