go函数默认值传递,无法修改外部变量;要修改必须显式传指针,在函数内用*解引用赋值,并注意判空防panic。

Go 函数里改不了外部变量?因为默认传值
Go 里所有参数都是值传递,func f(x int) 中的 x 是 int 的副本,改它不影响调用方的原始变量。想改外部变量,必须显式传它的地址——也就是指针。
怎么传指针?语法和常见错误
声明函数时参数类型写成指针类型,调用时用 & 取地址。最容易错的是:传了指针但函数里忘了用 * 解引用,或者传了值却误以为能改原变量。
-
正确写法:函数定义用
*int,函数内用*x = 42修改;调用时传&v -
典型错误:
func inc(x *int) { x = &newVal }—— 这只是改了局部指针变量x指向,没动原值 -
空指针 panic:如果传了
nil指针进去又直接*x = ...,运行时崩,得先判空
示例:
func setToTen(x *int) {
if x != nil {
*x = 10
}
}
v := 5
setToTen(&v)
// v 现在是 10
指针参数 vs 值参数:什么时候该用指针?
不是所有情况都得用指针。核心看两点:要不要修改原值、值本身大不大。
-
要改原值:比如
swap(a, b *int)、reset(buf *bytes.Buffer),必须指针 -
值很大(如大 struct):避免拷贝开销,即使不修改也常传
*MyStruct,但注意这不等于“允许修改”,得靠文档或命名约定说明意图 -
小基础类型(
int,bool,string):传值更清晰安全,除非明确需要修改
string 要特别注意:它是只读的底层字节数组描述符,传值不贵,且无法通过指针修改其内容(*string 只能换整个字符串,不是改里面字符)
立即学习“go语言免费学习笔记(深入)”;
结构体字段修改:指针接收者不是万能钥匙
给方法加指针接收者(func (s *MyStruct) SetX(v int)),是为了让方法能修改调用者的字段。但它和“传指针参数”是两回事——前者是调用方法时自动取地址,后者是手动传参。容易混淆的点是:
- 如果结构体变量是值类型(
var s MyStruct),调用s.SetX(1)会自动 &s;但如果它是接口类型或被赋值给interface{},可能丢失地址信息 - 如果接收者是值类型(
func (s MyStruct)),哪怕你传了指针过去,方法里改的仍是副本,外部不变 - 嵌套结构体字段赋值时,确保每一层指针都非 nil,否则
s.p.field = 1会 panic
真正要改外部变量,最直白的方式还是老老实实传指针参数——别指望编译器替你猜意图。
指针不是语法糖,是明确的内存操作信号。漏掉一个 *,或者多写一个 &,结果就完全不对。Go 不帮你隐式转换,这点得自己盯紧。










