以下代码展示了两个基准测试。第一个在每次迭代中通过值创建一个结构体,而第二个则使用指向结构体的指针。
为什么后者要慢20倍?我知道GoLang有垃圾回收问题,但逃逸分析不应该处理这些情况吗?
我正在使用go1.4beta1,但1.3.3给了我[同样-错误]的结果。
有任何想法吗?
为什么后者要慢20倍?我知道GoLang有垃圾回收问题,但逃逸分析不应该处理这些情况吗?
我正在使用go1.4beta1,但1.3.3给了我[同样-错误]的结果。
有任何想法吗?
package main
import "testing"
type Adder struct {
vals []int
}
func (a *Adder) add() int {
return a.vals[0] + a.vals[1]
}
func BenchmarkWithoutPointer(b *testing.B) {
accum := 0
for i := 0; i < b.N; i++ {
adder := Adder{[]int{accum, i}}
accum = adder.add()
}
_ = accum
}
func BenchmarkWithPointer(b *testing.B) {
accum := 0
for i := 0; i < b.N; i++ {
adder := &Adder{[]int{accum, i}}
accum = adder.add()
}
_ = accum
}
基准测试 go1.4.1:
$ go test -bench=.
testing: warning: no tests to run
PASS
BenchmarkWithoutPointer 1000000000 2.92 ns/op
BenchmarkWithPointer 30000000 57.8 ns/op
ok github.com/XXXXXXXXXX/bench/perf 5.010s
基准测试 go1.3.3:
testing: warning: no tests to run
PASS
BenchmarkWithoutPointer 500000000 7.89 ns/op
BenchmarkWithPointer 50000000 37.5 ns/op
ok
编辑:
结论:
正如Ainar-G所说,第二个基准测试中的[]int确实逃逸到了堆中。在阅读了有关1.4beta1的更多信息后,似乎新的GC计划导致访问堆时引入了新的写入障碍。但是原始执行似乎已经提高了。期待1.5 =)。