如何在Go中存储对操作结果的引用?

Rez*_*a S 7 dictionary pointers go

好吧用文字描述它很难,但是假设我有一个存储int指针的地图,并希望将操作的结果存储为我的哈希中的另一个键:

m := make(map[string]*int)

m["d"] = &(*m["x"] + *m["y"])
Run Code Online (Sandbox Code Playgroud)

这不起作用,并给我错误: cannot take the address of *m["x"] & *m["y"]

思考?

icz*_*cza 13

指针是内存地址.例如,变量在内存中具有地址.

操作的结果就像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"])
Run Code Online (Sandbox Code Playgroud)

输出(在Go Playground上试试):

0x10438300 3
Run Code Online (Sandbox Code Playgroud)

注意:如果上面的代码在函数中,d我们刚放入映射的局部变量()的地址将继续存活,即使我们从函数返回(即map返回或在外部创建 - 例如全局变量).在Go中,获取并返回局部变量的地址是完全安全的.编译器将分析代码,如果地址(指针)转义函数,它将自动在堆上分配(而不是在堆栈上).有关详细信息,请参阅常见问题解答:如何知道变量是在堆还是堆栈上分配?

注意#2:还有其他方法可以创建一个指向值的指针(详见答案:如何在Go中执行文字*int64?),但它们只是"技巧"而且不是更好或更有效.使用局部变量是最干净和推荐的方法.

例如,这也可以在不创建局部变量的情况下工作,但它显然根本不直观:

m["d"] = &[]int{*m["x"] + *m["y"]}[0]
Run Code Online (Sandbox Code Playgroud)

输出是一样的.在Go Playground尝试一下.