【GO语言】简单基准测试下,Go完全打不过Java,为什么?

起因,最近github上火了一个非常简单的测试项目,Go完全打不过JavaKotlin,且作者没有给.net的结果:
基准代码:bddicken/languages: Compare languages
测试结果:benjdd.com/languages/

十亿次双层嵌套循环测试结果

语言 耗时
C/clang -03 0.50s
Rust 0.50s
Java 0.54s
Kotlin 0.56s
Go 0.80s
Node.js 1.03s
Dart 1.34s
PyPy 1.53s
PHP 9.93s
Python 74.42s

递归斐波那契数列生成(S40=F40?1)结果

语言 耗时
C/clang -03 0.40s
Rust 0.41s
Java 0.48s
Kotlin 0.49s
Go 0.87s
Node.js 1.11s
Dart 0.78s
PyPy 2.75s
PHP 10.85s
Python 29.00s

测试环境

m3 mbp 16g

  • Clang version: Apple clang version 16.0.0 (clang-1600.0.26.4)
  • R version: Rscript (R) version 4.4.2 (2024-10-31)
  • Kotlin version: kotlinc-jvm 2.0.21 (JRE 23.0.1)
  • Java version: javac 23.0.1
  • Rust version: cargo 1.82.0
  • Node version: v22.11.0
  • Python version: 3.9.6
  • PHP version: 8.3.13
  • Dart version: 3.5.4
  • Go version: 1.21.2
  • PyPy: 7.3.17

循环测试Golang为什么慢,如何优化

package main // 基准测试代码
import (
    "fmt"
    "math/rand"
    "strconv"
    "os"
)

func main() {
  input, e := strconv.Atoi(os.Args[1]) // Get an input number from the command line
  if e != nil { panic(e) }
  u := int32(input)
  r := int32(rand.Intn(10000))           // Get a random number 0 <= r < 10k
  var a[10000]int32                      // Array of 10k elements initialized to 0
  for i := int32(0); i < 10000; i++ {         // 10k outer loop iterations
    for j := int32(0); j < 100000; j++ {      // 100k inner loop iterations, per outer loop iteration
      a[i] = a[i] + j%u                // Simple sum
    }
    a[i] += r                          // Add a random value to each element in array
  }
  fmt.Println(a[r])                    // Print out a single element from the array
}

问题的根源出在对a[i]的读写每次都需要去内存或缓存中进行,而老牌编译器会对这种嵌套循环中的变量创建一个临时变量,等内层循环结束,在写回原始变量,而临时变量更有可能被放入寄存器。如果我们把Go的代码稍微改一下,手动去做这段优化。

//...上文代码不变
  for i := int32(0); i < 10000; i++ {
    tmp := a[i]
    for j := int32(0); j < 100000; j++ {      // 100k inner loop iterations, per outer loop iteration
      tmp = tmp + j%u                // Simple sum
    }
    tmp += r                          // Add a random value to each element in array
    a[i] = tmp
  }
//...下文代码不变

具体测试结果我就不测了(电脑上OpenJDK都没装),可以看参考文献中:GoCN的文章

GO语言的递归为什么比不过Java

  • 栈空间处理
    • Go默认的栈空间较小(2KB-8KB),递归调用容易发生栈溢出
    • Java的栈空间默认更大(64KB-1MB),且可以轻松配置
    • Java的JVM对递归有专门的优化
  • 尾递归优化
    • Go编译器目前不支持尾递归优化
    • Java的HotSpot JVM会对热点代码进行尾递归优化
    • 这导致Go在深度递归时性能下降明显
  • JIT vs AOT
    • Java的JIT可以在运行时优化递归代码
    • Go是AOT编译,缺乏运行时优化机会
    • JIT能根据实际运行情况做更多优化

思考

其实我之前一直没想过,想当然的以为编译语言性能会高于Java这种需要依托类JVM运行的语言。事实上无论是多线程(当然Go语言也有优秀的三方协程池)还是基本逻辑代码等等,在很多方面Go都打不过Java,.net这种老牌编译工具链,这个github上的简单项目,不到两周的时间就一千个star,估计也是很多人没想到,在最简单的场景下,各种语言的表现和预期差别很大。虽然在业务上看,即便慢了一百倍的CPython在大多数场景也不会是性能瓶颈的根源,但作为一个程序人总要深思,不是么?

参考文献

bddicken/languages: Compare languages
benjdd.com/languages/
惊!Go在十亿次循环和百万任务中表现不如Java,究竟为何?

最后编辑于
?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容