带有 Go 泛型的单例模式
我正在尝试找出在 golang 中为泛型变量实现单例的最不糟糕的方法。使用普通的 sync.once
模式与全局变量是行不通的,因为通用类型信息在那里不可用(如下)。
这个示例是人为设计的,但实际上,维护单例的代码可以与定义 t
的客户端代码分开(例如在库中)。
假设此库代码,其中 t 的具体值未知:
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 }
并假设这个单独的客户端代码,其中 t
被定义为 string
被定义为 string
:
stringCache := getOrCreateCache[string]()
实现这一目标的最佳方法是什么?
正确答案
我最终使用 atomic.pointer
api 解决了这个问题,并将其整合到一个简单的库中,供其他感兴趣的人使用:单线。使用单线态重新处理原始帖子如下所示:
示例库代码:
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]{} }) }
客户端代码:
stringCache := getOrCreateCache[string]()
以及支持此功能的 singlet 库代码:
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 }
我希望对遇到这种情况的其他人有所帮助。
以上是带有 Go 泛型的单例模式的详细内容。更多信息请关注PHP中文网其他相关文章!

热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

记事本++7.3.1
好用且免费的代码编辑器

SublimeText3汉化版
中文版,非常好用

禅工作室 13.0.1
功能强大的PHP集成开发环境

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

热门话题

OpenSSL,作为广泛应用于安全通信的开源库,提供了加密算法、密钥和证书管理等功能。然而,其历史版本中存在一些已知安全漏洞,其中一些危害极大。本文将重点介绍Debian系统中OpenSSL的常见漏洞及应对措施。DebianOpenSSL已知漏洞:OpenSSL曾出现过多个严重漏洞,例如:心脏出血漏洞(CVE-2014-0160):该漏洞影响OpenSSL1.0.1至1.0.1f以及1.0.2至1.0.2beta版本。攻击者可利用此漏洞未经授权读取服务器上的敏感信息,包括加密密钥等。

本文演示了创建模拟和存根进行单元测试。 它强调使用接口,提供模拟实现的示例,并讨论最佳实践,例如保持模拟集中并使用断言库。 文章

本文探讨了GO的仿制药自定义类型约束。 它详细介绍了界面如何定义通用功能的最低类型要求,从而改善了类型的安全性和代码可重复使用性。 本文还讨论了局限性和最佳实践

本文讨论了GO中使用表驱动的测试,该方法使用测试用例表来测试具有多个输入和结果的功能。它突出了诸如提高的可读性,降低重复,可伸缩性,一致性和A

本文讨论了GO的反思软件包,用于运行时操作代码,对序列化,通用编程等有益。它警告性能成本,例如较慢的执行和更高的内存使用,建议明智的使用和最佳

本文使用跟踪工具探讨了GO应用程序执行流。 它讨论了手册和自动仪器技术,比较诸如Jaeger,Zipkin和Opentelemetry之类的工具,并突出显示有效的数据可视化
