Rumah > pembangunan bahagian belakang > Golang > Corak singleton dengan generik Go

Corak singleton dengan generik Go

WBOY
Lepaskan: 2024-02-05 22:54:07
ke hadapan
817 orang telah melayarinya

带有 Go 泛型的单例模式

Kandungan soalan

Saya cuba memikirkan cara paling tidak teruk untuk melaksanakan singleton bagi pembolehubah generik dalam golang. Menggunakan corak sync.once biasa dengan pembolehubah global tidak akan berfungsi kerana maklumat jenis biasa tidak tersedia di sana (di bawah).

Contoh ini dibuat-buat, tetapi pada hakikatnya, kod yang mengekalkan singleton boleh diasingkan daripada kod klien yang mentakrifkan t (cth. dalam perpustakaan).

Anggap kod perpustakaan ini, di mana nilai sebenar t tidak diketahui:

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
}
Salin selepas log masuk

Dan anggap kod pelanggan yang berasingan ini, dengan t 被定义为 string ditakrifkan sebagai rentetan:

stringCache := getOrCreateCache[string]()
Salin selepas log masuk

Apakah cara terbaik untuk mencapai matlamat ini?


Jawapan betul


Saya akhirnya menyelesaikan masalah ini menggunakan api atomic.pointer dan menyepadukannya ke dalam perpustakaan mudah untuk orang lain yang berminat: Satu pelapis. Mengolah semula jawatan asal menggunakan singlet kelihatan seperti ini:

Contoh kod perpustakaan:

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]{}
    })
}
Salin selepas log masuk

Kod pelanggan:

stringcache := getorcreatecache[string]()
Salin selepas log masuk

Dan singlet kod perpustakaan yang menyokong ciri ini:

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
}
Salin selepas log masuk

Saya harap ini membantu orang lain yang berada dalam situasi ini.

Atas ialah kandungan terperinci Corak singleton dengan generik Go. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

sumber:stackoverflow.com
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan