好吧用文字描述它很难,但是假设我有一个存储int
指针的地图,并希望将操作的结果存储为我的哈希中的另一个键:
m := make(map[string]*int) m["d"] = &(*m["x"] + *m["y"])
这不起作用,并给我错误: cannot take the address of *m["x"] & *m["y"]
思考?
指针是内存地址.例如,变量在内存中具有地址.
操作的结果就像3 + 4
没有地址一样,因为没有为它分配特定的内存.结果可能只存在于处理器寄存器中.
您必须分配可以放入地图的内存.最简单,最直接的是为它创建一个局部变量.
看这个例子:
x, y := 1, 2 m := map[string]*int{"x": &x, "y": &y} d := *m["x"] + *m["y"] m["d"] = &d fmt.Println(m["d"], *m["d"])
输出(在Go Playground上试试):
0x10438300 3
注意:如果上面的代码在函数中,d
我们刚放入映射的局部变量()的地址将继续存活,即使我们从函数返回(即map
返回或在外部创建 - 例如全局变量).在Go中,获取并返回局部变量的地址是完全安全的.编译器将分析代码,如果地址(指针)转义函数,它将自动在堆上分配(而不是在堆栈上).有关详细信息,请参阅常见问题解答:如何知道变量是在堆还是堆栈上分配?
注意#2:还有其他方法可以创建一个指向值的指针(详见答案:如何在Go中执行文字*int64?),但它们只是"技巧"而且不是更好或更有效.使用局部变量是最干净和推荐的方法.
例如,这也可以在不创建局部变量的情况下工作,但它显然根本不直观:
m["d"] = &[]int{*m["x"] + *m["y"]}[0]
输出是一样的.在Go Playground尝试一下.