以下代码显示了两个基准测试。第一个在每次迭代中按值创建一个结构,而第二个确实使用指向该结构的指针。为什么后者慢 20 倍?我知道 GoLang 的 GC 问题,但不应该逃避分析处理这些情况吗?我正在使用 go1.4beta1,但 1.3.3 给了我 [相同 - 错误]不同的结果。任何的想法 ?package mainimport "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 runPASSBenchmarkWithoutPointer 1000000000 2.92 ns/opBenchmarkWithPointer 30000000 57.8 ns/opok github.com/XXXXXXXXXX/bench/perf 5.010s基准 go1.3.3:testing: warning: no tests to runPASSBenchmarkWithoutPointer 500000000 7.89 ns/opBenchmarkWithPointer 50000000 37.5 ns/opok 编辑:结论:正如 Ainar-G 所说,[]int 确实在第二个基准测试中逃逸到堆。在阅读了更多关于 1.4beta1 的内容后,似乎在访问由新 GC 计划引起的堆时引入了新的写入障碍。但原始执行似乎有所增加。期待 1.5 =)。
1 回答
鸿蒙传说
TA贡献1865条经验 获得超7个赞
使用-mgcflag运行基准测试给出了可能的答案:
./main_test.go:16: BenchmarkWithoutPointer []int literal does not escape
(...)
./main_test.go:25: []int literal escapes to heap
您[]int在第二个示例中转义到堆,这比堆栈慢。如果您使用单独的x和y字段作为参数而不是切片
type Adder struct {
x, y int
}
func (a *Adder) add() int {
return a.x + a.y
}
基准显示了预期的行为:
BenchmarkWithoutPointer 1000000000 2.27 ns/op
BenchmarkWithPointer 2000000000 1.98 ns/op
- 1 回答
- 0 关注
- 180 浏览
添加回答
举报
0/150
提交
取消