Fix TypedValue

This commit is contained in:
wwqgtxx 2024-03-13 14:47:26 +08:00 committed by 世界
parent 807a51bb81
commit a069af4787
No known key found for this signature in database
GPG key ID: CD109927C34A63C4

View file

@ -10,26 +10,37 @@ type TypedValue[T any] struct {
value atomic.Value value atomic.Value
} }
// typedValue is a struct with determined type to resolve atomic.Value usages with interface types
// https://github.com/golang/go/issues/22550
//
// The intention to have an atomic value store for errors. However, running this code panics:
// panic: sync/atomic: store of inconsistently typed value into Value
// This is because atomic.Value requires that the underlying concrete type be the same (which is a reasonable expectation for its implementation).
// When going through the atomic.Value.Store method call, the fact that both these are of the error interface is lost.
type typedValue[T any] struct {
value T
}
func (t *TypedValue[T]) Load() T { func (t *TypedValue[T]) Load() T {
value := t.value.Load() value := t.value.Load()
if value == nil { if value == nil {
return common.DefaultValue[T]() return common.DefaultValue[T]()
} }
return value.(T) return value.(typedValue[T]).value
} }
func (t *TypedValue[T]) Store(value T) { func (t *TypedValue[T]) Store(value T) {
t.value.Store(value) t.value.Store(typedValue[T]{value})
} }
func (t *TypedValue[T]) Swap(new T) T { func (t *TypedValue[T]) Swap(new T) T {
old := t.value.Swap(new) old := t.value.Swap(typedValue[T]{new})
if old == nil { if old == nil {
return common.DefaultValue[T]() return common.DefaultValue[T]()
} }
return old.(T) return old.(typedValue[T]).value
} }
func (t *TypedValue[T]) CompareAndSwap(old, new T) bool { func (t *TypedValue[T]) CompareAndSwap(old, new T) bool {
return t.value.CompareAndSwap(old, new) return t.value.CompareAndSwap(typedValue[T]{old}, typedValue[T]{new})
} }