传值函数无法修改原始变量,因为Go中所有参数均为值传递,形参是实参的副本;要修改原始变量需用指针:函数声明int参数,调用时传&n,函数内用p赋值。

为什么传值函数无法修改原始变量
Go 中所有参数都是值传递,包括 int、string、struct 等值类型。这意味着函数内对形参的赋值操作只影响副本,不影响调用方的原始变量。
常见错误现象:
func addOne(x int) {
x = x + 1 // 这里改的是副本
}
n := 5
addOne(n)
fmt.Println(n) // 输出 5,不是 6
- 即使传入的是大 struct,Go 也会完整复制整个结构体内容
-
string虽然底层包含指针,但它是不可变类型,任何“修改”都生成新字符串,原变量不变 - 想让函数影响外部变量,必须显式传递该变量的地址(即指针)
如何用 *T 和 & 正确修改值
要修改原始变量,需在函数签名中声明指针类型参数,并在调用时用 & 取地址;函数体内用 * 解引用后赋值。
实操示例:
func increment(p *int) {
*p = *p + 1 // 解引用后修改原始内存位置的值
}
n := 10
increment(&n) // 传入 n 的地址
fmt.Println(n) // 输出 11
-
&n获取变量n在内存中的地址,类型是*int -
*p表示“取指针p所指向的值”,可读可写 - 如果传入
nil指针并解引用(如*p),会 panic:"invalid memory address or nil pointer dereference" - 不建议对常量或字面量取地址(如
&42),Go 编译器会报错
new(T) 和 &T{} 创建指针的区别
两者都能获得指向零值的指针,但语义和适用场景不同。
-
new(T)总是返回指向T类型零值的指针,适用于任意类型,包括基本类型和自定义类型 -
&T{}是取结构体/数组/切片字面量地址,只能用于复合类型(且要求字段可省略或全显式初始化) -
new(int)→*int指向0;&int{42}❌ 编译失败(不能对基本类型字面量取地址) -
new(MyStruct)和&MyStruct{}效果等价,但后者支持字段初始化:&MyStruct{Name: "a"}
指针与值类型在方法接收者中的实际影响
方法接收者是否用指针,直接决定能否修改调用者本身,也影响性能和接口实现行为。
立即学习“go语言免费学习笔记(深入)”;
- 值接收者方法:接收的是副本,无法修改原始结构体字段;每次调用都复制整个 struct
- 指针接收者方法:可修改字段,且避免复制开销;更重要的是,只有指针接收者方法能让类型满足需要指针接收者的方法集的接口
- 例如:
type User struct{ Name string },若只有func (u User) SetName(s string)(值接收者),则*User类型变量不能赋值给含SetName方法的接口变量 - 实践中,只要方法需修改字段,或 struct 较大(> 8 字节),就应使用指针接收者
真正容易被忽略的是:指针接收者方法可以被值类型变量调用(Go 自动取地址),但前提是该值是可寻址的——局部变量、切片元素、结构体字段可以;而字面量、函数返回值、map 值等不可寻址,此时调用指针接收者方法会编译失败。










