go - 将指向局部变量的指针分配给在Golang中作为指针返回的结构是否安全
问题描述
我发现 Golang 中的以下代码有效:
type user struct {
name *string
email *string
}
func foo() (*user, error) {
var myname string
var myemail string
// Some code here to fetch myname and myemail
u := user{}
u.name = &myname
u.email = &myemail
return &u, nil
}
我知道返回指向局部变量的指针是安全的,因为局部变量将在函数的范围内存活,所以我不担心&u
从函数返回foo()
。
我关心的是作业:
u.name = &myname
u.email = &myemail
u.name
Go 编译器在堆中分配和分配是偶然的,u.email
以便我能够在函数之外访问它,还是有一些保证它会一直工作(通过指针转义分析机制)?
如果上面的代码不安全,我会回退到以下内容:
u.name = new(string)
*u.name = myname
...
解决方案
我进一步深入挖掘并从这里和这里弄清楚如何进一步调查并满足我的查询,即在结构中设置指向局部变量的指针确实是安全的,该结构本身作为函数的指针返回。
我的程序(打印行号以便更好地理解):
1 package main
2
3 import "fmt"
4
5 type user struct {
6 Name *string
7 }
8
9 func setName() (*user) {
10 myname := "soumya"
11 u := user{}
12 u.Name = &myname
13 return &u
14 }
15
16 func main() {
17 v := setName()
18 fmt.Println(*v.Name)
19 }
输出go run -gcflags='-m -m'
:
$ go run -gcflags='-m -m' main.go
./main.go:9:6: can inline setName as: func() *user { myname := "soumya"; u := user literal; u.Name = &myname; return &u }
./main.go:16:6: cannot inline main: function too complex: cost 93 exceeds budget 80
./main.go:17:15: inlining call to setName func() *user { myname := "soumya"; u := user literal; u.Name = &myname; return &u }
./main.go:18:14: inlining call to fmt.Println func(...interface {}) (int, error) { return fmt.Fprintln(io.Writer(os.Stdout), fmt.a...) }
./main.go:13:10: &u escapes to heap
./main.go:13:10: from ~r0 (return) at ./main.go:13:3
./main.go:11:3: moved to heap: u
./main.go:12:12: &myname escapes to heap
./main.go:12:12: from u (dot-equals) at ./main.go:12:10
./main.go:12:12: from &u (address-of) at ./main.go:13:10
./main.go:12:12: from ~r0 (return) at ./main.go:13:3
./main.go:10:3: moved to heap: myname
./main.go:18:15: *v.Name escapes to heap
./main.go:18:15: from ~arg0 (assign-pair) at ./main.go:18:14
./main.go:18:14: io.Writer(os.Stdout) escapes to heap
./main.go:18:14: from io.Writer(os.Stdout) (passed to call[argument escapes]) at ./main.go:18:14
./main.go:17:15: main &myname does not escape
./main.go:17:15: main &u does not escape
./main.go:18:14: main []interface {} literal does not escape
<autogenerated>:1: os.(*File).close .this does not escape
soumya
输出状态&myname
转义到第 1 行的堆。12,因此它是安全的。此外,它在第号行之后不再转义。17.
推荐阅读
- python - Bot 在服务器上看不到成员
- mysql - MYSQL PRIMARY CHAR KEY 的最大最佳大小是多少
- android - 由于 Android 中的数据存储迁移,应用升级后应用崩溃
- python - 如何访问项目根文件夹中的文件?
- c++ - 在 C++ 中创建具有不同参数的多个构造函数的正确方法?
- python - matplotlib 中“关闭”路径参数的含义
- java - Java Annotation如何获取特定注解的当前ElemenType
- python - Pandas - 从 Pandas 中的同一字符串中选择几个浮点数来操作它们
- python - 用于动态分组的 Python pandas 小计,如何让总数始终出现在分组的末尾?
- node.js - 使用 Promise.all() 结果执行另一个查询的语法