Optimize Singleton (sometimes a simple lock is a better solution)

This commit is contained in:
Deluan 2023-12-27 22:12:34 -05:00
parent e50382e3bf
commit 0d8f8e3afd
2 changed files with 36 additions and 43 deletions

View file

@ -3,49 +3,42 @@ package singleton
import ( import (
"fmt" "fmt"
"reflect" "reflect"
"sync"
"github.com/navidrome/navidrome/log" "github.com/navidrome/navidrome/log"
) )
var ( var (
instances = make(map[string]any) instances = make(map[string]any)
getOrCreateC = make(chan entry) lock sync.RWMutex
) )
type entry struct {
f func() any
object any
resultC chan any
}
// GetInstance returns an existing instance of object. If it is not yet created, calls `constructor`, stores the // GetInstance returns an existing instance of object. If it is not yet created, calls `constructor`, stores the
// result for future calls and return it // result for future calls and return it
func GetInstance[T any](constructor func() T) T { func GetInstance[T any](constructor func() T) T {
var t T var v T
e := entry{ name := reflect.TypeOf(v).String()
object: t,
f: func() any { v, available := func() (T, bool) {
return constructor() lock.RLock()
}, defer lock.RUnlock()
resultC: make(chan any), v, available := instances[name].(T)
} return v, available
getOrCreateC <- e }()
v := <-e.resultC
return v.(T) if available {
return v
} }
func init() { lock.Lock()
go func() { defer lock.Unlock()
for { v, available = instances[name].(T)
e := <-getOrCreateC if available {
name := reflect.TypeOf(e.object).String() return v
v, created := instances[name] }
if !created {
v = e.f() v = constructor()
log.Trace("Created new singleton", "type", name, "instance", fmt.Sprintf("%+v", v)) log.Trace("Created new singleton", "type", name, "instance", fmt.Sprintf("%+v", v))
instances[name] = v instances[name] = v
} return v
e.resultC <- v
}
}()
} }

View file

@ -19,15 +19,15 @@ func TestSingleton(t *testing.T) {
var _ = Describe("GetInstance", func() { var _ = Describe("GetInstance", func() {
type T struct{ id string } type T struct{ id string }
var numInstances int var numInstancesCreated int
constructor := func() *T { constructor := func() *T {
numInstances++ numInstancesCreated++
return &T{id: uuid.NewString()} return &T{id: uuid.NewString()}
} }
It("calls the constructor to create a new instance", func() { It("calls the constructor to create a new instance", func() {
instance := singleton.GetInstance(constructor) instance := singleton.GetInstance(constructor)
Expect(numInstances).To(Equal(1)) Expect(numInstancesCreated).To(Equal(1))
Expect(instance).To(BeAssignableToTypeOf(&T{})) Expect(instance).To(BeAssignableToTypeOf(&T{}))
}) })
@ -36,24 +36,24 @@ var _ = Describe("GetInstance", func() {
newInstance := singleton.GetInstance(constructor) newInstance := singleton.GetInstance(constructor)
Expect(newInstance.id).To(Equal(instance.id)) Expect(newInstance.id).To(Equal(instance.id))
Expect(numInstances).To(Equal(1)) Expect(numInstancesCreated).To(Equal(1))
}) })
It("makes a distinction between a type and its pointer", func() { It("makes a distinction between a type and its pointer", func() {
instance := singleton.GetInstance(constructor) instance := singleton.GetInstance(constructor)
newInstance := singleton.GetInstance(func() T { newInstance := singleton.GetInstance(func() T {
numInstances++ numInstancesCreated++
return T{id: uuid.NewString()} return T{id: uuid.NewString()}
}) })
Expect(instance).To(BeAssignableToTypeOf(&T{})) Expect(instance).To(BeAssignableToTypeOf(&T{}))
Expect(newInstance).To(BeAssignableToTypeOf(T{})) Expect(newInstance).To(BeAssignableToTypeOf(T{}))
Expect(newInstance.id).ToNot(Equal(instance.id)) Expect(newInstance.id).ToNot(Equal(instance.id))
Expect(numInstances).To(Equal(2)) Expect(numInstancesCreated).To(Equal(2))
}) })
It("only calls the constructor once when called concurrently", func() { It("only calls the constructor once when called concurrently", func() {
const maxCalls = 8000 const maxCalls = 80000
var numCalls int32 var numCalls int32
start := sync.WaitGroup{} start := sync.WaitGroup{}
start.Add(1) start.Add(1)
@ -61,12 +61,12 @@ var _ = Describe("GetInstance", func() {
prepare.Add(maxCalls) prepare.Add(maxCalls)
done := sync.WaitGroup{} done := sync.WaitGroup{}
done.Add(maxCalls) done.Add(maxCalls)
numInstances = 0 numInstancesCreated = 0
for i := 0; i < maxCalls; i++ { for i := 0; i < maxCalls; i++ {
go func() { go func() {
start.Wait() start.Wait()
singleton.GetInstance(func() struct{ I int } { singleton.GetInstance(func() struct{ I int } {
numInstances++ numInstancesCreated++
return struct{ I int }{I: 1} return struct{ I int }{I: 1}
}) })
atomic.AddInt32(&numCalls, 1) atomic.AddInt32(&numCalls, 1)
@ -79,6 +79,6 @@ var _ = Describe("GetInstance", func() {
done.Wait() done.Wait()
Expect(numCalls).To(Equal(int32(maxCalls))) Expect(numCalls).To(Equal(int32(maxCalls)))
Expect(numInstances).To(Equal(1)) Expect(numInstancesCreated).To(Equal(1))
}) })
}) })