mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-05 05:27:37 +03:00
Optimize Singleton (sometimes a simple lock is a better solution)
This commit is contained in:
parent
e50382e3bf
commit
0d8f8e3afd
2 changed files with 36 additions and 43 deletions
|
@ -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
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -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))
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue