map[*rsa.PublicKey]*rsa.Privatekey 类型的映射有值,但 returns 为 nil
Map of type map[*rsa.PublicKey]*rsa.Privatekey has values, but returns nil
我有一个包含标题类型定义的结构。我创建了一个方法,它应该 return 地图中的一个值,但它变成了零。我怀疑这是因为它处理指针的方式,但我不确定它在哪里或如何出错。发生什么事了?
这是我为了查看发生了什么而添加的一些额外内容的片段:
func (s *stctClientPubKeys) GetServerPrivateKey(keyClientPubkey *rsa.PublicKey) *rsa.PrivateKey {
var keyPrivateKey *rsa.PrivateKey
s.RLock()
keyPrivateKey = s.mpClientPubKeys[keyClientPubkey]
s.RUnlock()
// TODO Remove this
fmt.Print("KEYINSTRUCT ")
fmt.Println(keyPrivateKey)
s.RLock()
fmt.Print("WHOLEMAPINSTRUCT ")
fmt.Println(s.mpClientPubKeys)
for k, v := range s.mpClientPubKeys {
chnLogging <- "GetServerPrivateKey(): Recv N = " + keyClientPubkey.N.String()
chnLogging <- "GetServerPrivateKey(): Key N = " + k.N.String()
chnLogging <- "GetServerPrivateKey(): Val N = " + v.PublicKey.N.String()
}
s.RUnlock()
return keyPrivateKey
}
示例中打印的内容 运行?
KEYINSTRUCT <nil>
WHOLEMAPINSTRUCT map[0xc00021e000:0xc0000644e0]
GetServerPrivateKey(): Recv N = 284429934903843367309<cut>
GetServerPrivateKey(): Key N = 284429934903843367309<cut>
GetServerPrivateKey(): Val N = 238865868438483289364<cut>
我尝试访问 returned 私钥时的最终结果当然是:
panic: runtime error: invalid memory address or nil pointer dereference
[signal SIGSEGV: segmentation violation code=0x1 addr=0x0 pc=0x53ce07]
正如 所暗示的,这可能是因为您在映射中存储了一个指针,而映射中的 public 键在语义上是等效的,但不是 ==
指数.
你没有说你是如何为你的函数获取参数的,但假设这是在某种服务器中,其中私钥是从线路或文件中解析的,你将得到一个新分配的对象。它会出现在内存中的其他地方,而不是地图中引用的指针,因此即使模数和指数相同,指针也会不同。该类型有一个Equal method可以用来比较,因为==
不行。
有两种简单的方法可以获得我怀疑您想要的行为:
- 使用切片并遍历,在每个切片上调用
.Equal(keyClientPubkey)
,而不是映射。这可能没问题,只要您的密钥少于(比方说)数百个 public 键即可。 (基准测试可以告诉您环境中的确切断点)
- 创建一个
toMapKey
辅助函数,它接受一个 *rsa.PublicKey
和 returns 可以用作键的可比较的东西(例如字符串)。一个简单的版本类似于 return fmt.Sprintf("n=%x|e=%x", pk.N, pk.E)
。使用它,您可以根据字符串表示形式存储和检索 public 键。
注意:如果这是一个安全敏感的代码路径(例如,如果对手可以随意触发它),我不会推荐上述任何一种方法,因为您可能会泄露计时信息。如果是这种情况,请向安全专家寻求帮助:)。如果您只是在启动或其他过程中这样做,可能没问题。
规范参考:
我有一个包含标题类型定义的结构。我创建了一个方法,它应该 return 地图中的一个值,但它变成了零。我怀疑这是因为它处理指针的方式,但我不确定它在哪里或如何出错。发生什么事了?
这是我为了查看发生了什么而添加的一些额外内容的片段:
func (s *stctClientPubKeys) GetServerPrivateKey(keyClientPubkey *rsa.PublicKey) *rsa.PrivateKey {
var keyPrivateKey *rsa.PrivateKey
s.RLock()
keyPrivateKey = s.mpClientPubKeys[keyClientPubkey]
s.RUnlock()
// TODO Remove this
fmt.Print("KEYINSTRUCT ")
fmt.Println(keyPrivateKey)
s.RLock()
fmt.Print("WHOLEMAPINSTRUCT ")
fmt.Println(s.mpClientPubKeys)
for k, v := range s.mpClientPubKeys {
chnLogging <- "GetServerPrivateKey(): Recv N = " + keyClientPubkey.N.String()
chnLogging <- "GetServerPrivateKey(): Key N = " + k.N.String()
chnLogging <- "GetServerPrivateKey(): Val N = " + v.PublicKey.N.String()
}
s.RUnlock()
return keyPrivateKey
}
示例中打印的内容 运行?
KEYINSTRUCT <nil>
WHOLEMAPINSTRUCT map[0xc00021e000:0xc0000644e0]
GetServerPrivateKey(): Recv N = 284429934903843367309<cut>
GetServerPrivateKey(): Key N = 284429934903843367309<cut>
GetServerPrivateKey(): Val N = 238865868438483289364<cut>
我尝试访问 returned 私钥时的最终结果当然是:
panic: runtime error: invalid memory address or nil pointer dereference
[signal SIGSEGV: segmentation violation code=0x1 addr=0x0 pc=0x53ce07]
正如 ==
指数.
你没有说你是如何为你的函数获取参数的,但假设这是在某种服务器中,其中私钥是从线路或文件中解析的,你将得到一个新分配的对象。它会出现在内存中的其他地方,而不是地图中引用的指针,因此即使模数和指数相同,指针也会不同。该类型有一个Equal method可以用来比较,因为==
不行。
有两种简单的方法可以获得我怀疑您想要的行为:
- 使用切片并遍历,在每个切片上调用
.Equal(keyClientPubkey)
,而不是映射。这可能没问题,只要您的密钥少于(比方说)数百个 public 键即可。 (基准测试可以告诉您环境中的确切断点) - 创建一个
toMapKey
辅助函数,它接受一个*rsa.PublicKey
和 returns 可以用作键的可比较的东西(例如字符串)。一个简单的版本类似于return fmt.Sprintf("n=%x|e=%x", pk.N, pk.E)
。使用它,您可以根据字符串表示形式存储和检索 public 键。
注意:如果这是一个安全敏感的代码路径(例如,如果对手可以随意触发它),我不会推荐上述任何一种方法,因为您可能会泄露计时信息。如果是这种情况,请向安全专家寻求帮助:)。如果您只是在启动或其他过程中这样做,可能没问题。
规范参考: