卓越飞翔博客卓越飞翔博客

卓越飞翔 - 您值得收藏的技术分享站
技术文章64334本站已运行4115

在 Go 中创建唯一映射键的最有效方法

在 go 中创建唯一映射键的最有效方法

在Go语言中,创建唯一映射键的最有效方法一直是开发者们关注的问题。在面对需要保证键的唯一性的场景时,我们需要寻找一种高效且可靠的方法。php小编百草将在本文中分享一种最有效的方法,帮助您在Go语言中创建唯一映射键,让您的代码更加优化和高效。无论是处理大规模数据还是实现高并发的应用,这些方法都能帮助您提高性能和效率。让我们一起来了解吧!

问题内容

我在图书馆的某个地方有一个 map[any]SomeType 。我希望库用户能够为该地图创建键,以便保证它们在单个应用程序执行中不会发生冲突,并且我希望这些键能够高效地进行地图查找。

首先想到的是使用某个唯一的空对象的内存地址。但到目前为止我尝试的一切都失败了:

package main

import "fmt"

var key1 = &struct{}{}
var key2 = &struct{}{}

var key3 = struct{}{}
var key4 = struct{}{}

var key5 = new(struct{})
var key6 = new(struct{})

func main() {
    fmt.Println("key1 == key2", key1 == key2)
    fmt.Println("key3 == key4", &key3 == &key4)
    fmt.Println("key5 == key6", key5 == key6)
    test(key1, key2, "func12")
    test(&key3, &key4, "func34")
    test(key5, key6, "func56")
}

func test(a, b any, msg string) {
    fmt.Println(msg, a == b)
}

打印

key1 == key2 true
key3 == key4 false
key5 == key6 true
func12 true
func34 true
func56 true

因此,获取空结构变量的地址几乎是可行的,直到将其传递给函数为止。然后区别就消失了。

我不想引入密钥注册表,因为它是一个不必要的复杂化。我也不想使用字符串,因为库的不同使用者需要协商密钥或使用名称空间,并且需要散列和比较字符串也是不必要的复杂化。

有没有我没想到的方法?

解决方法

标准库在使用 context.Context 时使用了一个“技巧”:上下文能够在其中携带任意值,并且这些值使用 interface{} 进行键控(从一段时间以来 any)1。然后,您自己的包可以为其将要使用的上下文键定义一个新的未导出类型,然后定义一组具有该类型的常量作为该包已知的上下文键。现在的技巧是,类型始终是 interface{} 类型的任何值的一部分,因此不可能创建与包的键冲突的接口值。

基本上是这样的:

package mypkg

type contextKey int

const (
  KeyFoo = contextKey(iota)
  KeyBar
)

现在,当您执行 key interface{} = KeyFoo 时,几乎可以保证程序中任何其他代码段都不能具有与 key 相同的值,因为其中一部分将(指向的内部指针)未导出输入 contextKey。 您可能需要阅读这篇经典文章来了解其工作原理(虽然有点生疏,但仍然有 99% 的正确性) )。

对我来说,这看起来是一种前进的方式:您的包的用户可以生成自己的密钥并将其提交到您的地图,该地图的密钥类型应为 interface{}any。无需由集中式注册表来移交这些密钥。

1 请参阅 context.Context.Value() 和 context.WithValue() 了解更多信息。后者提供了有关如何生成密钥的更多提示。

卓越飞翔博客
上一篇: `go build` 的输出标志 `-o` 有副作用吗?
下一篇: 我如何解组 JSON 数据并将其存储在 Go 中的结构中
留言与评论(共有 0 条评论)
   
验证码:
隐藏边栏