

新闻资讯
行业动态Go函数参数均为值传递,传值类型改副本不影响原变量,传指针类型可通过解引用修改原内存;结构体依大小和修改需求选传值或指针;切片/map/channel属“假引用”,可改元素但不可扩容;方法接收者依是否修改状态选择值或指针。
Go 里所有函数参数都是值传递,但“值”本身可能是整数,也可能是内存地址——关键就在这里。如果你发现 modifyName(p Person) 没改成功原始变量,大概率是因为你传了值;而换成 modifyName(p *Person) 并传入 &person,就能生效。
int、string、小 struct)传参:函数拿到的是副本,改了不影响原变量*int、*Person)传参:函数拿到的是地址副本,通过 *p 解引用就能写原内存一个 type User struct { Name string; Email string } 只有两个字段,传值没问题;但要是加了 AvatarData [1024 * 1024]byte,每次调用都复制 1MB,性能立刻掉下来。
func (u *User) Save()),整个类型建议统一用指针接收者,否则接口赋值会失败它们底层确实含指针(比如 slice 有 data *byte),所以你在函数里改 s[0] = 10 能看到效果;但想扩容或换底层数组?不行——因为传参仍是值传递,只复制了 header 结构。
s[0] = 10、m["k"] = v 都生效s = append(s, x) 不影响原切片;必须用 func extend(s *[]int, x int) + *s = append(*s, x)
Go 的方法集规则很严格:type T 的方法集只包含定义在 T 上的方法;*T 的方法集包含 T 和 *T 上的所有方法。这意味着:如果你写了 func (u *User) SetEmail(e string),那只有 *User 能调用它,User 值类型直接不满足接口。
立即学习“go语言免费学习笔记(深入)”;
Inc()、Reset()、SetXXX())→ 必须用指针接收者FullName()、IsValid())→ 值接收者更轻量、更安全*T,省得后期重构接口和调用点最容易被忽略的一点:不是“指针更高级”,而是“你传的值是否指向原始数据”。 写 fmt.Println(&x) 看地址,写 fmt.Println(*p) 看内容,两步拆开理解,比背概念管用得多。