如何在Golang中使用指针修改变量值_Golang指针操作与值类型对比

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

为什么传值函数无法修改原始变量

Go 中所有参数都是值传递,包括 intstringstruct 等值类型。这意味着函数内对形参的赋值操作只影响副本,不影响调用方的原始变量。

常见错误现象:

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"}

指针与值类型在方法接收者中的实际影响

方法接收者是否用指针,直接决定能否修改调用者本身,也影响性能和接口实现行为。

  • 值接收者方法:接收的是副本,无法修改原始结构体字段;每次调用都复制整个 struct
  • 指针接收者方法:可修改字段,且避免复制开销;更重要的是,只有指针接收者方法能让类型满足需要指针接收者的方法集的接口
  • 例如:type User struct{ Name string },若只有 func (u User) SetName(s string)(值接收者),则 *User 类型变量不能赋值给含 SetName 方法的接口变量
  • 实践中,只要方法需修改字段,或 struct 较大(> 8 字节),就应使用指针接收者

真正容易被忽略的是:指针接收者方法可以被值类型变量调用(Go 自动取地址),但前提是该值是可寻址的——局部变量、切片元素、结构体字段可以;而字面量、函数返回值、map 值等不可寻址,此时调用指针接收者方法会编译失败。