在Go中,指针赋值需确保对象可寻址且生命周期有效,不能对临时值或表达式取地址,如&(3+4)非法;函数返回值为临时对象,须先赋给变量再取地址;结构体字面量允许取地址,会创建持久实例;切片元素可取地址但扩容后指针可能失效,map元素禁止取地址以防扩容导致地址变化;值接收者方法操作副本,无法修改原值,应使用指针接收者以修改原始数据。

在Go语言中,将值类型赋值给指针时,本质是获取该值的内存地址并让指针指向它。这个过程看似简单,但有几个关键点需要注意,否则容易引发误解或潜在bug。
Go不允许对表达式产生的临时值直接取地址。例如,下面的写法是非法的:
错误示例:ptr := &(3 + 4) // 编译报错:cannot take the address of (3 + 4)
这是因为 3 + 4 是一个临时计算结果,没有固定的内存位置。只有可寻址的变量才能被取地址。
立即学习“go语言免费学习笔记(深入)”;
函数返回的是一个临时对象,不能对其取地址:
错误示例:
func getValue() int {
return 42
}
// ptr := &getValue() // 编译错误
如果需要指针,应先赋值给局部变量再取地址:
value := getValue()
ptr := &value // 正确
虽然结构体字面量看起来像临时值,但Go允许对它们取地址,因为这会创建一个在堆或栈上持久存在的实例:
type Person struct {
Name string
Age int
}
ptr := &Person{"Alice", 25} // 合法,等价于 new(Person) 然后赋值
这种写法很常见,特别是在初始化结构体指针时。
切片中的元素可以取地址,但要注意后续操作可能导致底层数组重分配,使原有指针失效:
slice := []int{1, 2, 3}
ptr := &slice[0]
slice = append(slice, 4) // 可能导致底层数组搬迁,ptr 指向旧内存,行为未定义
map中的元素则不能直接取地址:
m := map[string]int{"a": 1}; ptr := &m["a"] // 编译错误
原因是map元素的地址可能随着扩容而变化,Go禁止这种操作以保证安全。
当方法使用值接收者时,调用该方法的对象会被复制。若原对象是指针解引用而来,需注意副本与原值的关系:
type Data struct{ X int }
func (d Data) Set(x int) { d.X = x } // 值接收者,操作的是副本
d := &Data{10}
d.Set(20) // 不会改变 d 所指向的原始对象
若想修改原值,应使用指针接收者。
基本上就这些。关键是理解“可寻址性”和生命周期,避免对临时值或不可寻址对象取地址。Go的这些限制是为了防止悬空指针和内存错误。只要变量有确定的内存位置且生命周期足够长,就能安全地生成指针。不复杂但容易忽略细节。
以上就是golang值类型赋值给指针时的注意事项的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号