2010-12-03 05:34:57 +01:00
|
|
|
// Copyright 2009 The Go Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package sync_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
. "sync"
|
|
|
|
"testing"
|
|
|
|
)
|
|
|
|
|
|
|
|
type one int
|
|
|
|
|
|
|
|
func (o *one) Increment() {
|
|
|
|
*o++
|
|
|
|
}
|
|
|
|
|
2012-10-23 06:31:11 +02:00
|
|
|
func run(t *testing.T, once *Once, o *one, c chan bool) {
|
2010-12-03 05:34:57 +01:00
|
|
|
once.Do(func() { o.Increment() })
|
2012-10-23 06:31:11 +02:00
|
|
|
if v := *o; v != 1 {
|
|
|
|
t.Errorf("once failed inside run: %d is not 1", v)
|
|
|
|
}
|
2010-12-03 05:34:57 +01:00
|
|
|
c <- true
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOnce(t *testing.T) {
|
|
|
|
o := new(one)
|
|
|
|
once := new(Once)
|
|
|
|
c := make(chan bool)
|
|
|
|
const N = 10
|
|
|
|
for i := 0; i < N; i++ {
|
2012-10-23 06:31:11 +02:00
|
|
|
go run(t, once, o, c)
|
2010-12-03 05:34:57 +01:00
|
|
|
}
|
|
|
|
for i := 0; i < N; i++ {
|
|
|
|
<-c
|
|
|
|
}
|
|
|
|
if *o != 1 {
|
2012-10-23 06:31:11 +02:00
|
|
|
t.Errorf("once failed outside run: %d is not 1", *o)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOncePanic(t *testing.T) {
|
|
|
|
once := new(Once)
|
|
|
|
for i := 0; i < 2; i++ {
|
|
|
|
func() {
|
|
|
|
defer func() {
|
|
|
|
if recover() == nil {
|
|
|
|
t.Fatalf("Once.Do() has not panic'ed")
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
once.Do(func() {
|
|
|
|
panic("failed")
|
|
|
|
})
|
|
|
|
}()
|
2010-12-03 05:34:57 +01:00
|
|
|
}
|
2012-10-23 06:31:11 +02:00
|
|
|
once.Do(func() {})
|
|
|
|
once.Do(func() {
|
|
|
|
t.Fatalf("Once called twice")
|
|
|
|
})
|
2010-12-03 05:34:57 +01:00
|
|
|
}
|
2011-09-16 17:47:21 +02:00
|
|
|
|
|
|
|
func BenchmarkOnce(b *testing.B) {
|
|
|
|
var once Once
|
|
|
|
f := func() {}
|
2014-07-19 10:53:52 +02:00
|
|
|
b.RunParallel(func(pb *testing.PB) {
|
|
|
|
for pb.Next() {
|
|
|
|
once.Do(f)
|
|
|
|
}
|
|
|
|
})
|
2011-09-16 17:47:21 +02:00
|
|
|
}
|