Maison > développement back-end > Golang > le corps du texte

Modèle Singleton avec les génériques Go

WBOY
Libérer: 2024-02-05 22:54:07
avant
729 Les gens l'ont consulté

带有 Go 泛型的单例模式

Contenu de la question

J'essaie de trouver la façon la moins pire d'implémenter un singleton pour une variable générique dans Golang. Utiliser le modèle plain sync.once avec des variables globales ne fonctionnera pas car les informations de type commun n'y sont pas disponibles (ci-dessous).

Cet exemple est artificiel, mais en réalité, le code qui maintient le singleton peut être séparé du code client qui définit t (par exemple dans une bibliothèque).

Supposons ce code de bibliothèque, où la valeur exacte de t est inconnue :

type cache[t any] struct{}

var (
    cacheonce sync.once
    cache     cache[any] // global singleton
)

func getorcreatecache[t any]() cache[t] {
    cacheonce.do(func() {
        typedcache := buildcache()
        cache = typedcache.(cache[any]) // invalid type assertion
    })
    return cache.(cache[t]) // invalid type assertion
}
Copier après la connexion

Et supposons ce code client distinct, où t 被定义为 string est défini comme string :

stringCache := getOrCreateCache[string]()
Copier après la connexion

Quelle est la meilleure façon d’y parvenir ?


Bonne réponse


J'ai fini par résoudre ce problème en utilisant l'API atomic.pointer et je l'ai intégré dans une bibliothèque simple pour les autres personnes intéressées : One-liner. Retravailler le message original à l'aide de singlets ressemble à ceci :

Exemple de code de bibliothèque :

type cache[t any] struct{}

var singleton = &singlet.singleton{}

func getorcreatecache[t any]() (cache[t], err) {
    return singlet.getordo(singleton, func() cache[t] {
        return cache[t]{}
    })
}
Copier après la connexion

Code client :

stringcache := getorcreatecache[string]()
Copier après la connexion

Et le code de la bibliothèque singlet qui prend en charge cette fonctionnalité :

var ErrTypeMismatch = errors.New("the requested type does not match the singleton type")

type Singleton struct {
    p   atomic.Pointer[any]
    mtx sync.Mutex
}

func GetOrDo[T any](singleton *Singleton, fn func() T) (result T, err error) {
    maybeResult := singleton.p.Load()
    if maybeResult == nil {
        // Lock to guard against applying fn twice
        singleton.mtx.Lock()
        defer singleton.mtx.Unlock()
        maybeResult = singleton.p.Load()

        // Double check
        if maybeResult == nil {
            result = fn()
            var resultAny any = result
            singleton.p.Store(&resultAny)
            return result, nil
        }
    }

    var ok bool
    result, ok = (*maybeResult).(T)
    if !ok {
        return *new(T), ErrTypeMismatch
    }
    return result, nil
}
Copier après la connexion

J'espère que cela aidera les autres qui sont dans cette situation.

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

source:stackoverflow.com
Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn
Tutoriels populaires
Plus>
Derniers téléchargements
Plus>
effets Web
Code source du site Web
Matériel du site Web
Modèle frontal
À propos de nous Clause de non-responsabilité Sitemap
Site Web PHP chinois:Formation PHP en ligne sur le bien-être public,Aidez les apprenants PHP à grandir rapidement!